Phase 1: Station shell - /station route replaces /pos (with redirect) - Shared lock screen, activity tracking, auto-lock timer - Permission-gated tab bar (POS | Repairs | Lessons) - POSRegister embedded mode (skip lock/timer/topbar) - Sidebar navigates to /station Phase 2: Repairs station — front desk - Status bar with count badges per status group, clickable filters - Ticket queue panel with search and status filtering - Ticket detail panel with status progress, notes, photos, line items - Full stepped intake form: customer → item → problem/estimate → review - Service template quick-add for line items Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
159 lines
4.6 KiB
TypeScript
159 lines
4.6 KiB
TypeScript
import { useEffect, useRef, useCallback } from 'react'
|
|
import { useQuery } from '@tanstack/react-query'
|
|
import { queryOptions } from '@tanstack/react-query'
|
|
import { api } from '@/lib/api-client'
|
|
import { usePOSStore } from '@/stores/pos.store'
|
|
import { currentDrawerOptions, transactionOptions } from '@/api/pos'
|
|
import { POSTopBar } from './pos-top-bar'
|
|
import { POSItemPanel } from './pos-item-panel'
|
|
import { POSCartPanel } from './pos-cart-panel'
|
|
import { POSLockScreen } from './pos-lock-screen'
|
|
|
|
interface Location {
|
|
id: string
|
|
name: string
|
|
}
|
|
|
|
interface AppConfigEntry {
|
|
key: string
|
|
value: string | null
|
|
}
|
|
|
|
function locationsOptions() {
|
|
return queryOptions({
|
|
queryKey: ['locations'],
|
|
queryFn: () => api.get<{ data: Location[] }>('/v1/locations'),
|
|
})
|
|
}
|
|
|
|
function configOptions(key: string) {
|
|
return queryOptions({
|
|
queryKey: ['config', key],
|
|
queryFn: async (): Promise<string | null> => {
|
|
try {
|
|
const entry = await api.get<AppConfigEntry>(`/v1/config/${key}`)
|
|
return entry.value
|
|
} catch {
|
|
return null
|
|
}
|
|
},
|
|
})
|
|
}
|
|
|
|
interface POSRegisterProps {
|
|
embedded?: boolean
|
|
}
|
|
|
|
export function POSRegister({ embedded }: POSRegisterProps = {}) {
|
|
const { locationId, setLocation, currentTransactionId, setDrawerSession, locked, lock, touchActivity, token } = usePOSStore()
|
|
|
|
// Fetch lock timeout from config (standalone only)
|
|
const { data: lockTimeoutStr } = useQuery({
|
|
...configOptions('pos_lock_timeout'),
|
|
enabled: !!token && !embedded,
|
|
})
|
|
const lockTimeoutMinutes = parseInt(lockTimeoutStr ?? '15') || 15
|
|
|
|
// Auto-lock timer (standalone only — station shell handles this when embedded)
|
|
const timerRef = useRef<ReturnType<typeof setInterval> | null>(null)
|
|
|
|
useEffect(() => {
|
|
if (embedded) return
|
|
if (locked || lockTimeoutMinutes === 0) {
|
|
if (timerRef.current) clearInterval(timerRef.current)
|
|
return
|
|
}
|
|
|
|
timerRef.current = setInterval(() => {
|
|
const { lastActivity, locked: isLocked } = usePOSStore.getState()
|
|
if (!isLocked && Date.now() - lastActivity > lockTimeoutMinutes * 60_000) {
|
|
lock()
|
|
}
|
|
}, 30_000)
|
|
|
|
return () => {
|
|
if (timerRef.current) clearInterval(timerRef.current)
|
|
}
|
|
}, [embedded, locked, lockTimeoutMinutes, lock])
|
|
|
|
// Track activity (standalone only)
|
|
const handleActivity = useCallback(() => {
|
|
if (!embedded && !locked) touchActivity()
|
|
}, [embedded, locked, touchActivity])
|
|
|
|
// Fetch locations (standalone only — station shell handles this when embedded)
|
|
const { data: locationsData } = useQuery({
|
|
...locationsOptions(),
|
|
enabled: !!token && !embedded,
|
|
})
|
|
const locations = locationsData?.data ?? []
|
|
|
|
// Auto-select first location (standalone only)
|
|
useEffect(() => {
|
|
if (embedded) return
|
|
if (!locationId && locations.length > 0) {
|
|
setLocation(locations[0].id)
|
|
}
|
|
}, [embedded, locationId, locations, setLocation])
|
|
|
|
// Fetch current drawer for selected location
|
|
const { data: drawer } = useQuery({
|
|
...currentDrawerOptions(locationId),
|
|
retry: false,
|
|
enabled: !!locationId && !!token,
|
|
})
|
|
|
|
// Sync drawer session ID
|
|
useEffect(() => {
|
|
if (drawer?.id && drawer.status === 'open') {
|
|
setDrawerSession(drawer.id)
|
|
} else {
|
|
setDrawerSession(null)
|
|
}
|
|
}, [drawer, setDrawerSession])
|
|
|
|
// Fetch current transaction
|
|
const { data: transaction } = useQuery({
|
|
...transactionOptions(currentTransactionId),
|
|
enabled: !!currentTransactionId && !!token,
|
|
})
|
|
|
|
// Embedded mode: just the content panels, no wrapper/lock/topbar
|
|
if (embedded) {
|
|
return (
|
|
<div className="flex flex-1 h-full min-h-0">
|
|
<div className="w-[60%] border-r border-border overflow-hidden">
|
|
<POSItemPanel transaction={transaction ?? null} />
|
|
</div>
|
|
<div className="w-[40%] overflow-hidden">
|
|
<POSCartPanel transaction={transaction ?? null} />
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
return (
|
|
<div
|
|
className="relative flex flex-col h-full"
|
|
onPointerDown={handleActivity}
|
|
onKeyDown={handleActivity}
|
|
>
|
|
{locked && <POSLockScreen />}
|
|
<POSTopBar
|
|
locations={locations}
|
|
locationId={locationId}
|
|
onLocationChange={setLocation}
|
|
drawer={drawer ?? null}
|
|
/>
|
|
<div className="flex flex-1 min-h-0">
|
|
<div className="w-[60%] border-r border-border overflow-hidden">
|
|
<POSItemPanel transaction={transaction ?? null} />
|
|
</div>
|
|
<div className="w-[40%] overflow-hidden">
|
|
<POSCartPanel transaction={transaction ?? null} />
|
|
</div>
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|