| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733 |
- import { Outlet, Link, useLocation } from 'react-router-dom'
- import { useEffect, useState, useRef, useCallback } from 'react'
- import { toast } from 'sonner'
- import {
- Power,
- RotateCcw,
- RefreshCw,
- Loader2,
- CheckCircle,
- Terminal,
- Copy,
- Eye,
- EyeOff,
- X,
- Menu,
- FileText,
- Download,
- HardDriveDownload,
- Grid,
- ListMusic,
- SlidersHorizontal,
- Lightbulb,
- Settings as SettingsIcon,
- Wifi,
- WifiOff,
- AlertCircle,
- Home,
- Sun,
- Moon,
- ChevronDown,
- PlayCircle,
- StopCircle,
- } from 'lucide-react'
- import { NowPlayingBar } from '@/components/NowPlayingBar'
- import { Button } from '@/components/ui/button'
- import { Popover, PopoverContent, PopoverTrigger } from '@/components/ui/popover'
- import { Separator } from '@/components/ui/separator'
- import { cacheAllPreviews } from '@/lib/previewCache'
- import { TableSelector } from '@/components/TableSelector'
- import { useTable } from '@/contexts/TableContext'
- import { apiClient } from '@/lib/apiClient'
- import ShinyText from '@/components/ShinyText'
- const navItems = [
- { path: '/', label: 'Browse', icon: Grid, title: 'Browse Patterns' },
- { path: '/playlists', label: 'Playlists', icon: ListMusic, title: 'Playlists' },
- { path: '/table-control', label: 'Control', icon: SlidersHorizontal, title: 'Table Control' },
- { path: '/led', label: 'LED', icon: Lightbulb, title: 'LED Control' },
- { path: '/settings', label: 'Settings', icon: SettingsIcon, title: 'Settings' },
- ]
- const DEFAULT_APP_NAME = 'Dune Weaver'
- export function Layout() {
- const location = useLocation()
- // Scroll to top on route change
- useEffect(() => {
- window.scrollTo(0, 0)
- }, [location.pathname])
- // Multi-table context - must be called before any hooks that depend on activeTable
- const { activeTable, tables } = useTable()
- // Use table name as app name when multiple tables exist
- const hasMultipleTables = tables.length > 1
- const [isDark, setIsDark] = useState(() => {
- if (typeof window !== 'undefined') {
- const saved = localStorage.getItem('theme')
- if (saved) return saved === 'dark'
- return window.matchMedia('(prefers-color-scheme: dark)').matches
- }
- return false
- })
- // App customization
- const [appName, setAppName] = useState(DEFAULT_APP_NAME)
- const [customLogo, setCustomLogo] = useState<string | null>(null)
- // Display name: when multiple tables exist, use the active table's name; otherwise use app settings
- // Get the table from the tables array (most up-to-date source) to ensure we have current data
- const activeTableData = tables.find(t => t.id === activeTable?.id)
- const tableName = activeTableData?.name || activeTable?.name
- const displayName = hasMultipleTables && tableName ? tableName : appName
- // Connection status
- const [isConnected, setIsConnected] = useState(false)
- const [isBackendConnected, setIsBackendConnected] = useState(false)
- const [isHoming, setIsHoming] = useState(false)
- const [homingDismissed, setHomingDismissed] = useState(false)
- const [homingJustCompleted, setHomingJustCompleted] = useState(false)
- const [homingCountdown, setHomingCountdown] = useState(0)
- const [keepHomingLogsOpen, setKeepHomingLogsOpen] = useState(false)
- const wasHomingRef = useRef(false)
- const [connectionAttempts, setConnectionAttempts] = useState(0)
- const wsRef = useRef<WebSocket | null>(null)
- // Sensor homing failure state
- const [sensorHomingFailed, setSensorHomingFailed] = useState(false)
- const [isRecoveringHoming, setIsRecoveringHoming] = useState(false)
- // Fetch app settings
- const fetchAppSettings = () => {
- apiClient.get<{ app?: { name?: string; custom_logo?: string } }>('/api/settings')
- .then((settings) => {
- if (settings.app?.name) {
- setAppName(settings.app.name)
- } else {
- setAppName(DEFAULT_APP_NAME)
- }
- setCustomLogo(settings.app?.custom_logo || null)
- })
- .catch(() => {})
- }
- useEffect(() => {
- fetchAppSettings()
- // Listen for branding updates from Settings page
- const handleBrandingUpdate = () => {
- fetchAppSettings()
- }
- window.addEventListener('branding-updated', handleBrandingUpdate)
- return () => {
- window.removeEventListener('branding-updated', handleBrandingUpdate)
- }
- // Refetch when active table changes
- }, [activeTable?.id])
- // Homing completion countdown timer
- useEffect(() => {
- if (!homingJustCompleted || keepHomingLogsOpen) return
- if (homingCountdown <= 0) {
- // Countdown finished, dismiss the overlay
- setHomingJustCompleted(false)
- setKeepHomingLogsOpen(false)
- return
- }
- const timer = setTimeout(() => {
- setHomingCountdown((prev) => prev - 1)
- }, 1000)
- return () => clearTimeout(timer)
- }, [homingJustCompleted, homingCountdown, keepHomingLogsOpen])
- // Mobile menu state
- const [isMobileMenuOpen, setIsMobileMenuOpen] = useState(false)
- // Logs drawer state
- const [isLogsOpen, setIsLogsOpen] = useState(false)
- const [logsDrawerHeight, setLogsDrawerHeight] = useState(256) // Default 256px (h-64)
- const [isResizing, setIsResizing] = useState(false)
- const isResizingRef = useRef(false)
- const startYRef = useRef(0)
- const startHeightRef = useRef(0)
- // Handle drawer resize
- const handleResizeStart = (e: React.MouseEvent | React.TouchEvent) => {
- e.preventDefault()
- isResizingRef.current = true
- setIsResizing(true)
- startYRef.current = 'touches' in e ? e.touches[0].clientY : e.clientY
- startHeightRef.current = logsDrawerHeight
- document.body.style.cursor = 'ns-resize'
- document.body.style.userSelect = 'none'
- }
- useEffect(() => {
- const handleResizeMove = (e: MouseEvent | TouchEvent) => {
- if (!isResizingRef.current) return
- const clientY = 'touches' in e ? e.touches[0].clientY : e.clientY
- const delta = startYRef.current - clientY
- const newHeight = Math.min(Math.max(startHeightRef.current + delta, 150), window.innerHeight - 150)
- setLogsDrawerHeight(newHeight)
- }
- const handleResizeEnd = () => {
- if (isResizingRef.current) {
- isResizingRef.current = false
- setIsResizing(false)
- document.body.style.cursor = ''
- document.body.style.userSelect = ''
- }
- }
- window.addEventListener('mousemove', handleResizeMove)
- window.addEventListener('mouseup', handleResizeEnd)
- window.addEventListener('touchmove', handleResizeMove)
- window.addEventListener('touchend', handleResizeEnd)
- return () => {
- window.removeEventListener('mousemove', handleResizeMove)
- window.removeEventListener('mouseup', handleResizeEnd)
- window.removeEventListener('touchmove', handleResizeMove)
- window.removeEventListener('touchend', handleResizeEnd)
- }
- }, [])
- // Now Playing bar state
- const [isNowPlayingOpen, setIsNowPlayingOpen] = useState(false)
- const [openNowPlayingExpanded, setOpenNowPlayingExpanded] = useState(false)
- const [currentPlayingFile, setCurrentPlayingFile] = useState<string | null>(null) // Track current file for header button
- const wasPlayingRef = useRef<boolean | null>(null) // Track previous playing state (null = first message)
- // Derive isCurrentlyPlaying from currentPlayingFile
- const isCurrentlyPlaying = Boolean(currentPlayingFile)
- // Listen for playback-started event (dispatched when user starts a pattern)
- useEffect(() => {
- const handlePlaybackStarted = () => {
- setIsNowPlayingOpen(true)
- setOpenNowPlayingExpanded(true)
- setIsLogsOpen(false)
- // Reset expanded flag after animation
- setTimeout(() => setOpenNowPlayingExpanded(false), 500)
- }
- window.addEventListener('playback-started', handlePlaybackStarted)
- return () => window.removeEventListener('playback-started', handlePlaybackStarted)
- }, [])
- const [logs, setLogs] = useState<Array<{ timestamp: string; level: string; logger: string; message: string }>>([])
- const [logLevelFilter, setLogLevelFilter] = useState<string>('ALL')
- const [logsTotal, setLogsTotal] = useState(0)
- const [logsHasMore, setLogsHasMore] = useState(false)
- const [isLoadingMoreLogs, setIsLoadingMoreLogs] = useState(false)
- const logsWsRef = useRef<WebSocket | null>(null)
- const logsContainerRef = useRef<HTMLDivElement>(null)
- const logsLoadedCountRef = useRef(0) // Track how many logs we've loaded (for offset)
- // Check device connection status via WebSocket
- // This effect runs once on mount and manages its own reconnection logic
- useEffect(() => {
- let reconnectTimeout: ReturnType<typeof setTimeout> | null = null
- let isMounted = true
- const connectWebSocket = () => {
- if (!isMounted) return
- // Only close existing connection if it's open (not still connecting)
- // This prevents "WebSocket closed before connection established" errors
- if (wsRef.current) {
- if (wsRef.current.readyState === WebSocket.OPEN) {
- wsRef.current.close()
- wsRef.current = null
- } else if (wsRef.current.readyState === WebSocket.CONNECTING) {
- // Already connecting, don't interrupt
- return
- }
- }
- const ws = new WebSocket(apiClient.getWebSocketUrl('/ws/status'))
- // Assign to ref IMMEDIATELY so concurrent calls see it's connecting
- wsRef.current = ws
- ws.onopen = () => {
- if (!isMounted) {
- // Component unmounted while connecting - close the WebSocket now
- ws.close()
- return
- }
- setIsBackendConnected(true)
- setConnectionAttempts(0)
- // Dispatch event so pages can refetch data
- window.dispatchEvent(new CustomEvent('backend-connected'))
- }
- ws.onmessage = (event) => {
- if (!isMounted) return
- try {
- const data = JSON.parse(event.data)
- // Handle status updates
- if (data.type === 'status_update' && data.data) {
- // Update device connection status from the status message
- if (data.data.connection_status !== undefined) {
- setIsConnected(data.data.connection_status)
- }
- // Update homing status and detect completion
- if (data.data.is_homing !== undefined) {
- const newIsHoming = data.data.is_homing
- // Detect transition from not homing to homing - reset dismissal
- if (!wasHomingRef.current && newIsHoming) {
- setHomingDismissed(false)
- }
- // Detect transition from homing to not homing
- if (wasHomingRef.current && !newIsHoming) {
- // Homing just completed - show completion state with countdown
- // But not if sensor homing failed (that shows a different dialog)
- if (!data.data.sensor_homing_failed) {
- setHomingJustCompleted(true)
- setHomingCountdown(5)
- setHomingDismissed(false)
- }
- }
- wasHomingRef.current = newIsHoming
- setIsHoming(newIsHoming)
- }
- // Update sensor homing failure status
- if (data.data.sensor_homing_failed !== undefined) {
- setSensorHomingFailed(data.data.sensor_homing_failed)
- }
- // Auto-open/close Now Playing bar based on playback state
- // Track current file - this is the most reliable indicator of playback
- const currentFile = data.data.current_file || null
- setCurrentPlayingFile(currentFile)
- const isPlaying = Boolean(currentFile) || Boolean(data.data.is_running) || Boolean(data.data.is_paused)
- // Skip auto-open on first message (page refresh) - only react to state changes
- if (wasPlayingRef.current !== null) {
- if (isPlaying && !wasPlayingRef.current) {
- // Playback just started - open the Now Playing bar in expanded mode
- setIsNowPlayingOpen(true)
- setOpenNowPlayingExpanded(true)
- // Close the logs drawer if open
- setIsLogsOpen(false)
- // Reset the expanded flag after a short delay
- setTimeout(() => setOpenNowPlayingExpanded(false), 500)
- // Dispatch event so pages can close their sidebars/panels
- window.dispatchEvent(new CustomEvent('playback-started'))
- } else if (!isPlaying && wasPlayingRef.current) {
- // Playback just stopped - close the Now Playing bar
- setIsNowPlayingOpen(false)
- }
- }
- wasPlayingRef.current = isPlaying
- }
- } catch {
- // Ignore parse errors
- }
- }
- ws.onclose = () => {
- if (!isMounted) return
- wsRef.current = null
- setIsBackendConnected(false)
- setConnectionAttempts((prev) => prev + 1)
- // Reconnect after 3 seconds (don't change device status on WS disconnect)
- reconnectTimeout = setTimeout(connectWebSocket, 3000)
- }
- ws.onerror = () => {
- if (!isMounted) return
- setIsBackendConnected(false)
- }
- }
- // Reset playing state on mount
- wasPlayingRef.current = null
- // Connect on mount
- connectWebSocket()
- // Subscribe to base URL changes (when user switches tables)
- // This triggers reconnection to the new backend
- const unsubscribe = apiClient.onBaseUrlChange(() => {
- if (isMounted) {
- wasPlayingRef.current = null // Reset playing state for new table
- setCurrentPlayingFile(null) // Reset playback state for new table
- setIsConnected(false) // Reset connection status until new table reports
- setIsBackendConnected(false) // Show connecting state
- setSensorHomingFailed(false) // Reset sensor homing failure state for new table
- connectWebSocket()
- }
- })
- return () => {
- isMounted = false
- unsubscribe()
- if (reconnectTimeout) {
- clearTimeout(reconnectTimeout)
- }
- if (wsRef.current) {
- // Only close if already OPEN - CONNECTING WebSockets will close in onopen
- if (wsRef.current.readyState === WebSocket.OPEN) {
- wsRef.current.close()
- }
- wsRef.current = null
- }
- }
- }, []) // Empty deps - runs once on mount, reconnects via apiClient listener
- // Connect to logs WebSocket when drawer opens
- useEffect(() => {
- if (!isLogsOpen) {
- // Close WebSocket when drawer closes - only if OPEN (CONNECTING will close in onopen)
- if (logsWsRef.current && logsWsRef.current.readyState === WebSocket.OPEN) {
- logsWsRef.current.close()
- }
- logsWsRef.current = null
- return
- }
- let shouldConnect = true
- // Fetch initial logs (most recent)
- const fetchInitialLogs = async () => {
- try {
- type LogEntry = { timestamp: string; level: string; logger: string; message: string }
- type LogsResponse = { logs: LogEntry[]; total: number; has_more: boolean }
- const data = await apiClient.get<LogsResponse>('/api/logs?limit=200')
- // Filter out empty/invalid log entries
- const validLogs = (data.logs || []).filter(
- (log) => log && log.message && log.message.trim() !== ''
- )
- // API returns newest first, reverse to show oldest first (newest at bottom)
- setLogs(validLogs.reverse())
- setLogsTotal(data.total || 0)
- setLogsHasMore(data.has_more || false)
- logsLoadedCountRef.current = validLogs.length
- // Scroll to bottom after initial load
- setTimeout(() => {
- if (logsContainerRef.current) {
- logsContainerRef.current.scrollTop = logsContainerRef.current.scrollHeight
- }
- }, 100)
- } catch {
- // Ignore errors
- }
- }
- fetchInitialLogs()
- // Connect to WebSocket for real-time updates
- let reconnectTimeout: ReturnType<typeof setTimeout> | null = null
- const connectLogsWebSocket = () => {
- // Don't interrupt an existing connection that's still connecting
- if (logsWsRef.current) {
- if (logsWsRef.current.readyState === WebSocket.CONNECTING) {
- return // Already connecting, wait for it
- }
- if (logsWsRef.current.readyState === WebSocket.OPEN) {
- logsWsRef.current.close()
- }
- logsWsRef.current = null
- }
- const ws = new WebSocket(apiClient.getWebSocketUrl('/ws/logs'))
- // Assign to ref IMMEDIATELY so concurrent calls see it's connecting
- logsWsRef.current = ws
- ws.onopen = () => {
- if (!shouldConnect) {
- // Effect cleanup ran while connecting - close now
- ws.close()
- return
- }
- console.log('Logs WebSocket connected')
- }
- ws.onmessage = (event) => {
- try {
- const message = JSON.parse(event.data)
- // Skip heartbeat messages
- if (message.type === 'heartbeat') {
- return
- }
- // Extract log from wrapped structure
- const log = message.type === 'log_entry' ? message.data : message
- // Skip empty or invalid log entries
- if (!log || !log.message || log.message.trim() === '') {
- return
- }
- // Append new log - no limit, lazy loading handles old logs
- setLogs((prev) => [...prev, log])
- // Auto-scroll to bottom if user is near the bottom
- setTimeout(() => {
- if (logsContainerRef.current) {
- const { scrollTop, scrollHeight, clientHeight } = logsContainerRef.current
- // Only auto-scroll if user is within 100px of the bottom
- if (scrollHeight - scrollTop - clientHeight < 100) {
- logsContainerRef.current.scrollTop = scrollHeight
- }
- }
- }, 10)
- } catch {
- // Ignore parse errors
- }
- }
- ws.onclose = () => {
- if (!shouldConnect) return
- console.log('Logs WebSocket closed, reconnecting...')
- // Reconnect after 3 seconds if drawer is still open
- reconnectTimeout = setTimeout(() => {
- if (shouldConnect && logsWsRef.current === ws) {
- connectLogsWebSocket()
- }
- }, 3000)
- }
- ws.onerror = (error) => {
- console.error('Logs WebSocket error:', error)
- }
- }
- connectLogsWebSocket()
- return () => {
- shouldConnect = false
- if (reconnectTimeout) {
- clearTimeout(reconnectTimeout)
- }
- if (logsWsRef.current) {
- // Only close if already OPEN - CONNECTING WebSockets will close in onopen
- if (logsWsRef.current.readyState === WebSocket.OPEN) {
- logsWsRef.current.close()
- }
- logsWsRef.current = null
- }
- }
- // Also reconnect when active table changes
- }, [isLogsOpen, activeTable?.id])
- // Load older logs when user scrolls to top (lazy loading)
- const loadOlderLogs = useCallback(async () => {
- if (isLoadingMoreLogs || !logsHasMore) return
- setIsLoadingMoreLogs(true)
- try {
- type LogEntry = { timestamp: string; level: string; logger: string; message: string }
- type LogsResponse = { logs: LogEntry[]; total: number; has_more: boolean }
- const offset = logsLoadedCountRef.current
- const data = await apiClient.get<LogsResponse>(`/api/logs?limit=100&offset=${offset}`)
- const validLogs = (data.logs || []).filter(
- (log) => log && log.message && log.message.trim() !== ''
- )
- if (validLogs.length > 0) {
- // Prepend older logs (they come newest-first, so reverse them)
- setLogs((prev) => [...validLogs.reverse(), ...prev])
- logsLoadedCountRef.current += validLogs.length
- setLogsHasMore(data.has_more || false)
- setLogsTotal(data.total || 0)
- // Maintain scroll position after prepending
- setTimeout(() => {
- if (logsContainerRef.current) {
- // Calculate approximate height of new content (rough estimate: 24px per log line)
- const newContentHeight = validLogs.length * 24
- logsContainerRef.current.scrollTop = newContentHeight
- }
- }, 10)
- } else {
- setLogsHasMore(false)
- }
- } catch {
- // Ignore errors
- } finally {
- setIsLoadingMoreLogs(false)
- }
- }, [isLoadingMoreLogs, logsHasMore])
- // Scroll event handler for lazy loading
- useEffect(() => {
- const container = logsContainerRef.current
- if (!container || !isLogsOpen) return
- const handleScroll = () => {
- // Load more when scrolled to top (within 50px)
- if (container.scrollTop < 50 && logsHasMore && !isLoadingMoreLogs) {
- loadOlderLogs()
- }
- }
- container.addEventListener('scroll', handleScroll)
- return () => container.removeEventListener('scroll', handleScroll)
- }, [isLogsOpen, logsHasMore, isLoadingMoreLogs, loadOlderLogs])
- const handleToggleLogs = () => {
- setIsLogsOpen((prev) => !prev)
- }
- // Filter logs by level
- const filteredLogs = logLevelFilter === 'ALL'
- ? logs
- : logs.filter((log) => log.level === logLevelFilter)
- // Format timestamp safely
- const formatTimestamp = (timestamp: string) => {
- if (!timestamp) return '--:--:--'
- try {
- const date = new Date(timestamp)
- if (isNaN(date.getTime())) return '--:--:--'
- return date.toLocaleTimeString()
- } catch {
- return '--:--:--'
- }
- }
- // Copy logs to clipboard (with fallback for non-HTTPS)
- const handleCopyLogs = () => {
- const text = filteredLogs
- .map((log) => `${formatTimestamp(log.timestamp)} [${log.level}] ${log.message}`)
- .join('\n')
- copyToClipboard(text)
- }
- // Helper to copy text with fallback for non-secure contexts
- const copyToClipboard = (text: string) => {
- if (navigator.clipboard && window.isSecureContext) {
- navigator.clipboard.writeText(text).then(() => {
- toast.success('Logs copied to clipboard')
- }).catch(() => {
- toast.error('Failed to copy logs')
- })
- } else {
- // Fallback for non-secure contexts (http://)
- const textArea = document.createElement('textarea')
- textArea.value = text
- textArea.style.position = 'fixed'
- textArea.style.left = '-9999px'
- document.body.appendChild(textArea)
- textArea.select()
- try {
- document.execCommand('copy')
- toast.success('Logs copied to clipboard')
- } catch {
- toast.error('Failed to copy logs')
- }
- document.body.removeChild(textArea)
- }
- }
- // Download logs as file
- const handleDownloadLogs = () => {
- const text = filteredLogs
- .map((log) => `${log.timestamp} [${log.level}] [${log.logger}] ${log.message}`)
- .join('\n')
- const blob = new Blob([text], { type: 'text/plain' })
- const url = URL.createObjectURL(blob)
- const a = document.createElement('a')
- a.href = url
- a.download = `dune-weaver-logs-${new Date().toISOString().split('T')[0]}.txt`
- a.click()
- URL.revokeObjectURL(url)
- }
- const handleRestart = async () => {
- if (!confirm('Are you sure you want to restart Docker containers?')) return
- try {
- await apiClient.post('/api/system/restart')
- toast.success('Docker containers are restarting...')
- } catch {
- toast.error('Failed to restart Docker containers')
- }
- }
- const handleShutdown = async () => {
- if (!confirm('Are you sure you want to shutdown the system?')) return
- try {
- await apiClient.post('/api/system/shutdown')
- toast.success('System is shutting down...')
- } catch {
- toast.error('Failed to shutdown system')
- }
- }
- // Handle sensor homing recovery
- const handleSensorHomingRecovery = async (switchToCrashHoming: boolean) => {
- setIsRecoveringHoming(true)
- try {
- const response = await apiClient.post<{
- success: boolean
- sensor_homing_failed?: boolean
- message?: string
- }>('/recover_sensor_homing', {
- switch_to_crash_homing: switchToCrashHoming
- })
- if (response.success) {
- toast.success(response.message || 'Homing completed successfully')
- setSensorHomingFailed(false)
- } else if (response.sensor_homing_failed) {
- // Sensor homing failed again
- toast.error(response.message || 'Sensor homing failed again')
- } else {
- toast.error(response.message || 'Recovery failed')
- }
- } catch {
- toast.error('Failed to recover from sensor homing failure')
- } finally {
- setIsRecoveringHoming(false)
- }
- }
- // Update document title based on current page
- useEffect(() => {
- const currentNav = navItems.find((item) => item.path === location.pathname)
- if (currentNav) {
- document.title = `${currentNav.title} | ${displayName}`
- } else {
- document.title = displayName
- }
- }, [location.pathname, displayName])
- useEffect(() => {
- if (isDark) {
- document.documentElement.classList.add('dark')
- localStorage.setItem('theme', 'dark')
- } else {
- document.documentElement.classList.remove('dark')
- localStorage.setItem('theme', 'light')
- }
- }, [isDark])
- // Blocking overlay logs state - shows connection attempts
- const [connectionLogs, setConnectionLogs] = useState<Array<{ timestamp: string; level: string; message: string }>>([])
- const blockingLogsRef = useRef<HTMLDivElement>(null)
- // Cache progress state
- const [cacheProgress, setCacheProgress] = useState<{
- is_running: boolean
- stage: string
- processed_files: number
- total_files: number
- current_file: string
- error?: string
- } | null>(null)
- const cacheWsRef = useRef<WebSocket | null>(null)
- // Cache All Previews prompt state
- const [showCacheAllPrompt, setShowCacheAllPrompt] = useState(false)
- const [cacheAllProgress, setCacheAllProgress] = useState<{
- inProgress: boolean
- completed: number
- total: number
- done: boolean
- } | null>(null)
- // Blocking overlay logs WebSocket ref
- const blockingLogsWsRef = useRef<WebSocket | null>(null)
- // Add connection/homing logs when overlay is shown
- useEffect(() => {
- const showOverlay = !isBackendConnected || isHoming || homingJustCompleted
- if (!showOverlay) {
- setConnectionLogs([])
- // Close WebSocket if open - only if OPEN (CONNECTING will close in onopen)
- if (blockingLogsWsRef.current && blockingLogsWsRef.current.readyState === WebSocket.OPEN) {
- blockingLogsWsRef.current.close()
- }
- blockingLogsWsRef.current = null
- return
- }
- // Don't clear logs or reconnect WebSocket during completion state
- if (homingJustCompleted && !isHoming) {
- return
- }
- // Add log entry helper
- const addLog = (level: string, message: string, timestamp?: string) => {
- setConnectionLogs((prev) => {
- const newLog = {
- timestamp: timestamp || new Date().toISOString(),
- level,
- message,
- }
- const newLogs = [...prev, newLog].slice(-100) // Keep last 100 entries
- return newLogs
- })
- // Auto-scroll to bottom
- setTimeout(() => {
- if (blockingLogsRef.current) {
- blockingLogsRef.current.scrollTop = blockingLogsRef.current.scrollHeight
- }
- }, 10)
- }
- // If homing, connect to logs WebSocket to stream real logs
- if (isHoming && isBackendConnected) {
- addLog('INFO', 'Homing started...')
- let shouldConnect = true
- // Don't interrupt an existing connection that's still connecting
- if (blockingLogsWsRef.current) {
- if (blockingLogsWsRef.current.readyState === WebSocket.CONNECTING) {
- return // Already connecting, wait for it
- }
- if (blockingLogsWsRef.current.readyState === WebSocket.OPEN) {
- blockingLogsWsRef.current.close()
- }
- blockingLogsWsRef.current = null
- }
- const ws = new WebSocket(apiClient.getWebSocketUrl('/ws/logs'))
- // Assign to ref IMMEDIATELY so concurrent calls see it's connecting
- blockingLogsWsRef.current = ws
- ws.onopen = () => {
- if (!shouldConnect) {
- // Effect cleanup ran while connecting - close now
- ws.close()
- }
- }
- ws.onmessage = (event) => {
- try {
- const message = JSON.parse(event.data)
- if (message.type === 'heartbeat') return
- const log = message.type === 'log_entry' ? message.data : message
- if (!log || !log.message || log.message.trim() === '') return
- // Filter for homing-related logs
- const msg = log.message.toLowerCase()
- const isHomingLog =
- msg.includes('homing') ||
- msg.includes('home') ||
- msg.includes('$h') ||
- msg.includes('idle') ||
- msg.includes('unlock') ||
- msg.includes('alarm') ||
- msg.includes('grbl') ||
- msg.includes('connect') ||
- msg.includes('serial') ||
- msg.includes('device') ||
- msg.includes('position') ||
- msg.includes('zeroing') ||
- msg.includes('movement') ||
- log.logger?.includes('connection')
- if (isHomingLog) {
- addLog(log.level, log.message, log.timestamp)
- }
- } catch {
- // Ignore parse errors
- }
- }
- return () => {
- shouldConnect = false
- // Only close if already OPEN - CONNECTING WebSockets will close in onopen
- if (ws.readyState === WebSocket.OPEN) {
- ws.close()
- }
- blockingLogsWsRef.current = null
- }
- }
- // If backend disconnected, show connection retry logs
- if (!isBackendConnected) {
- addLog('INFO', `Attempting to connect to backend at ${window.location.host}...`)
- const interval = setInterval(() => {
- addLog('INFO', `Retrying connection to WebSocket /ws/status...`)
- apiClient.get('/api/settings')
- .then(() => {
- addLog('INFO', 'HTTP endpoint responding, waiting for WebSocket...')
- })
- .catch(() => {
- // Still down
- })
- }, 3000)
- return () => clearInterval(interval)
- }
- }, [isBackendConnected, isHoming, homingJustCompleted])
- // Cache progress WebSocket connection - always connected to monitor cache generation
- useEffect(() => {
- if (!isBackendConnected) return
- let reconnectTimeout: ReturnType<typeof setTimeout> | null = null
- let shouldConnect = true
- const connectCacheWebSocket = () => {
- if (!shouldConnect) return
- // Don't interrupt an existing connection that's still connecting
- if (cacheWsRef.current) {
- if (cacheWsRef.current.readyState === WebSocket.CONNECTING) {
- return // Already connecting, wait for it
- }
- if (cacheWsRef.current.readyState === WebSocket.OPEN) {
- return // Already connected
- }
- // CLOSING or CLOSED state - clear the ref
- cacheWsRef.current = null
- }
- const ws = new WebSocket(apiClient.getWebSocketUrl('/ws/cache-progress'))
- // Assign to ref IMMEDIATELY so concurrent calls see it's connecting
- cacheWsRef.current = ws
- ws.onopen = () => {
- if (!shouldConnect) {
- // Effect cleanup ran while connecting - close now
- ws.close()
- }
- }
- ws.onmessage = (event) => {
- try {
- const message = JSON.parse(event.data)
- if (message.type === 'cache_progress') {
- const data = message.data
- if (data.is_running) {
- // Cache generation is running - show splash screen
- setCacheProgress(data)
- } else if (data.stage === 'complete') {
- // Cache generation just completed
- if (cacheProgress?.is_running) {
- // Was running before, now complete - show cache all prompt
- const promptShown = localStorage.getItem('cacheAllPromptShown')
- if (!promptShown) {
- setTimeout(() => {
- setCacheAllProgress(null) // Reset to clean state
- setShowCacheAllPrompt(true)
- }, 500)
- }
- }
- setCacheProgress(null)
- } else {
- // Not running and not complete (idle state)
- setCacheProgress(null)
- }
- }
- } catch {
- // Ignore parse errors
- }
- }
- ws.onclose = () => {
- if (!shouldConnect) return
- cacheWsRef.current = null
- // Reconnect after 3 seconds
- if (shouldConnect && isBackendConnected) {
- reconnectTimeout = setTimeout(connectCacheWebSocket, 3000)
- }
- }
- ws.onerror = () => {
- // Will trigger onclose
- }
- }
- connectCacheWebSocket()
- return () => {
- shouldConnect = false
- if (reconnectTimeout) {
- clearTimeout(reconnectTimeout)
- }
- if (cacheWsRef.current) {
- // Only close if already OPEN - CONNECTING WebSockets will close in onopen
- if (cacheWsRef.current.readyState === WebSocket.OPEN) {
- cacheWsRef.current.close()
- }
- cacheWsRef.current = null
- }
- }
- }, [isBackendConnected]) // Only reconnect based on backend connection, not cache state
- // Calculate cache progress percentage
- const cachePercentage = cacheProgress?.total_files
- ? Math.round((cacheProgress.processed_files / cacheProgress.total_files) * 100)
- : 0
- const getCacheStageText = () => {
- if (!cacheProgress) return ''
- switch (cacheProgress.stage) {
- case 'starting':
- return 'Initializing...'
- case 'metadata':
- return 'Processing pattern metadata'
- case 'images':
- return 'Generating pattern previews'
- default:
- return 'Processing...'
- }
- }
- // Cache all previews in browser using IndexedDB
- const handleCacheAllPreviews = async () => {
- setCacheAllProgress({ inProgress: true, completed: 0, total: 0, done: false })
- const result = await cacheAllPreviews((progress) => {
- setCacheAllProgress({ inProgress: !progress.done, ...progress })
- })
- if (result.success) {
- if (result.cached === 0) {
- toast.success('All patterns are already cached!')
- } else {
- toast.success(`Cached ${result.cached} pattern previews`)
- }
- } else {
- setCacheAllProgress(null)
- toast.error('Failed to cache previews')
- }
- }
- const handleSkipCacheAll = () => {
- localStorage.setItem('cacheAllPromptShown', 'true')
- setShowCacheAllPrompt(false)
- setCacheAllProgress(null)
- }
- const handleCloseCacheAllDone = () => {
- localStorage.setItem('cacheAllPromptShown', 'true')
- setShowCacheAllPrompt(false)
- setCacheAllProgress(null)
- }
- const cacheAllPercentage = cacheAllProgress?.total
- ? Math.round((cacheAllProgress.completed / cacheAllProgress.total) * 100)
- : 0
- return (
- <div className="min-h-dvh bg-background flex flex-col">
- {/* Sensor Homing Failure Popup */}
- {sensorHomingFailed && (
- <div className="fixed inset-0 z-50 bg-black/50 backdrop-blur-sm flex items-center justify-center p-4">
- <div className="bg-background rounded-lg shadow-xl w-full max-w-md border border-destructive/30">
- <div className="p-6">
- <div className="text-center space-y-4">
- <div className="inline-flex items-center justify-center w-16 h-16 rounded-full bg-destructive/10 mb-2">
- <AlertCircle className="h-10 w-10 text-destructive" />
- </div>
- <h2 className="text-xl font-semibold">Sensor Homing Failed</h2>
- <p className="text-muted-foreground text-sm">
- The sensor homing process could not complete. The limit sensors may not be positioned correctly or may be malfunctioning.
- </p>
- <div className="bg-amber-500/10 border border-amber-500/20 p-3 rounded-lg text-sm text-left">
- <p className="text-amber-600 dark:text-amber-400 font-medium mb-2">
- Troubleshooting steps:
- </p>
- <ul className="text-amber-600 dark:text-amber-400 space-y-1 list-disc list-inside">
- <li>Check that the limit sensors are properly connected</li>
- <li>Verify the sensor positions are correct</li>
- <li>Ensure nothing is blocking the sensor path</li>
- <li>Check for loose wiring connections</li>
- </ul>
- </div>
- <p className="text-muted-foreground text-sm">
- Connection will not be established until this is resolved.
- </p>
- {/* Action Buttons */}
- {!isRecoveringHoming ? (
- <div className="flex flex-col gap-2 pt-2">
- <Button
- onClick={() => handleSensorHomingRecovery(false)}
- className="w-full gap-2"
- >
- <RefreshCw className="h-4 w-4" />
- Retry Sensor Homing
- </Button>
- <Button
- variant="secondary"
- onClick={() => handleSensorHomingRecovery(true)}
- className="w-full gap-2"
- >
- <RefreshCw className="h-4 w-4" />
- Switch to Crash Homing
- </Button>
- <p className="text-xs text-muted-foreground">
- Crash homing moves the arm to a physical stop without using sensors.
- </p>
- </div>
- ) : (
- <div className="flex items-center justify-center gap-2 py-4">
- <Loader2 className="h-5 w-5 text-primary animate-spin" />
- <span className="text-muted-foreground">Attempting recovery...</span>
- </div>
- )}
- </div>
- </div>
- </div>
- </div>
- )}
- {/* Cache Progress Blocking Overlay */}
- {cacheProgress?.is_running && (
- <div className="fixed inset-0 z-50 bg-background/95 backdrop-blur-sm flex flex-col items-center justify-center p-4">
- <div className="w-full max-w-md space-y-6">
- <div className="text-center space-y-4">
- <div className="inline-flex items-center justify-center w-16 h-16 rounded-full bg-primary/10 mb-2">
- <RefreshCw className="h-10 w-10 text-primary animate-pulse" />
- </div>
- <h2 className="text-2xl font-bold">Initializing Pattern Cache</h2>
- <p className="text-muted-foreground">
- Preparing your pattern previews...
- </p>
- </div>
- {/* Progress Bar */}
- <div className="space-y-2">
- <div className="w-full bg-muted rounded-full h-2 overflow-hidden">
- <div
- className="bg-primary h-2 rounded-full transition-all duration-300"
- style={{ width: `${cachePercentage}%` }}
- />
- </div>
- <div className="flex justify-between text-sm text-muted-foreground">
- <span>
- {cacheProgress.processed_files} of {cacheProgress.total_files} patterns
- </span>
- <span>{cachePercentage}%</span>
- </div>
- </div>
- {/* Stage Info */}
- <div className="text-center space-y-1">
- <p className="text-sm font-medium">{getCacheStageText()}</p>
- {cacheProgress.current_file && (
- <p className="text-xs text-muted-foreground truncate max-w-full">
- {cacheProgress.current_file}
- </p>
- )}
- </div>
- {/* Hint */}
- <p className="text-center text-xs text-muted-foreground">
- This only happens once after updates or when new patterns are added
- </p>
- </div>
- </div>
- )}
- {/* Cache All Previews Prompt Modal */}
- {showCacheAllPrompt && (
- <div className="fixed inset-0 z-50 bg-black/50 backdrop-blur-sm flex items-center justify-center p-4">
- <div className="bg-background rounded-lg shadow-xl w-full max-w-md">
- <div className="p-6">
- <div className="text-center space-y-4">
- <div className="inline-flex items-center justify-center w-12 h-12 rounded-full bg-primary/10 mb-2">
- <HardDriveDownload className="h-6 w-6 text-primary" />
- </div>
- <h2 className="text-xl font-semibold">Cache All Pattern Previews?</h2>
- <p className="text-muted-foreground text-sm">
- Would you like to cache all pattern previews for faster browsing? This will download and store preview images in your browser for instant loading.
- </p>
- <div className="bg-amber-500/10 border border-amber-500/20 p-3 rounded-lg text-sm">
- <p className="text-amber-600 dark:text-amber-400">
- <strong>Note:</strong> This cache is browser-specific. You'll need to repeat this for each browser you use.
- </p>
- </div>
- {/* Initial state - show buttons */}
- {!cacheAllProgress && (
- <div className="flex gap-3 justify-center">
- <Button variant="ghost" onClick={handleSkipCacheAll}>
- Skip for now
- </Button>
- <Button variant="secondary" onClick={handleCacheAllPreviews} className="gap-2">
- <RefreshCw className="h-5 w-5" />
- Cache All
- </Button>
- </div>
- )}
- {/* Progress section */}
- {cacheAllProgress && !cacheAllProgress.done && (
- <div className="space-y-2">
- <div className="w-full bg-muted rounded-full h-2 overflow-hidden">
- <div
- className="bg-primary h-2 rounded-full transition-all duration-300"
- style={{ width: `${cacheAllPercentage}%` }}
- />
- </div>
- <div className="flex justify-between text-sm text-muted-foreground">
- <span>
- {cacheAllProgress.completed} of {cacheAllProgress.total} previews
- </span>
- <span>{cacheAllPercentage}%</span>
- </div>
- </div>
- )}
- {/* Completion message */}
- {cacheAllProgress?.done && (
- <div className="space-y-4">
- <p className="text-green-600 dark:text-green-400 flex items-center justify-center gap-2">
- <CheckCircle className="h-4 w-4" />
- All {cacheAllProgress.total} previews cached successfully!
- </p>
- <Button onClick={handleCloseCacheAllDone} className="w-full">
- Done
- </Button>
- </div>
- )}
- </div>
- </div>
- </div>
- </div>
- )}
- {/* Backend Connection / Homing Blocking Overlay */}
- {/* Don't show this overlay when sensor homing failed - that has its own dialog */}
- {!sensorHomingFailed && (!isBackendConnected || (isHoming && !homingDismissed) || homingJustCompleted) && (
- <div className="fixed inset-0 z-50 bg-background/95 backdrop-blur-sm flex flex-col items-center justify-center p-4">
- <div className="w-full max-w-2xl space-y-6">
- {/* Status Header */}
- <div className="text-center space-y-4">
- <div className={`inline-flex items-center justify-center w-16 h-16 rounded-full mb-2 ${
- homingJustCompleted
- ? 'bg-green-500/10'
- : isHoming
- ? 'bg-primary/10'
- : 'bg-amber-500/10'
- }`}>
- {homingJustCompleted ? (
- <CheckCircle className="h-10 w-10 text-green-500" />
- ) : isHoming ? (
- <Loader2 className="h-10 w-10 text-primary animate-spin" />
- ) : (
- <Home className="h-10 w-10 text-amber-500 animate-pulse" />
- )}
- </div>
- <h2 className="text-2xl font-bold">
- {homingJustCompleted
- ? 'Homing Complete'
- : isHoming
- ? 'Homing in Progress'
- : 'Connecting to Backend'
- }
- </h2>
- <p className="text-muted-foreground">
- {homingJustCompleted
- ? 'Table is ready to use'
- : isHoming
- ? 'Moving to home position... This may take up to 90 seconds.'
- : connectionAttempts === 0
- ? 'Establishing connection...'
- : `Reconnecting... (attempt ${connectionAttempts})`
- }
- </p>
- <div className="flex items-center justify-center gap-2 text-sm text-muted-foreground">
- <span className={`w-2 h-2 rounded-full ${
- homingJustCompleted
- ? 'bg-green-500'
- : isHoming
- ? 'bg-primary animate-pulse'
- : 'bg-amber-500 animate-pulse'
- }`} />
- <span>
- {homingJustCompleted
- ? keepHomingLogsOpen
- ? 'Viewing logs'
- : `Closing in ${homingCountdown}s...`
- : isHoming
- ? 'Do not interrupt the homing process'
- : `Waiting for server at ${window.location.host}`
- }
- </span>
- </div>
- </div>
- {/* Logs Panel */}
- <div className="bg-muted/50 rounded-lg border overflow-hidden">
- <div className="flex items-center justify-between px-4 py-2 border-b bg-muted">
- <div className="flex items-center gap-2">
- <Terminal className="h-4 w-4" />
- <span className="text-sm font-medium">
- {isHoming || homingJustCompleted ? 'Homing Log' : 'Connection Log'}
- </span>
- </div>
- <div className="flex items-center gap-2">
- <button
- onClick={() => {
- const logText = connectionLogs
- .map((log) => `[${new Date(log.timestamp).toLocaleTimeString()}] [${log.level}] ${log.message}`)
- .join('\n')
- copyToClipboard(logText)
- }}
- className="text-xs text-muted-foreground hover:text-foreground flex items-center gap-1 transition-colors"
- title="Copy logs to clipboard"
- >
- <Copy className="h-3.5 w-3.5" />
- Copy
- </button>
- <span className="text-xs text-muted-foreground">
- {connectionLogs.length} entries
- </span>
- </div>
- </div>
- <div
- ref={blockingLogsRef}
- className="h-48 overflow-auto p-3 font-mono text-xs space-y-0.5"
- >
- {connectionLogs.map((log, i) => (
- <div key={i} className="py-0.5 flex gap-2">
- <span className="text-muted-foreground shrink-0">
- {formatTimestamp(log.timestamp)}
- </span>
- <span className={`shrink-0 font-semibold ${
- log.level === 'ERROR' ? 'text-red-500' :
- log.level === 'WARNING' ? 'text-amber-500' :
- log.level === 'DEBUG' ? 'text-muted-foreground' :
- 'text-foreground'
- }`}>
- [{log.level}]
- </span>
- <span className="break-all">{log.message}</span>
- </div>
- ))}
- </div>
- </div>
- {/* Action buttons for homing completion */}
- {homingJustCompleted && (
- <div className="flex justify-center gap-3">
- {!keepHomingLogsOpen ? (
- <>
- <Button
- variant="secondary"
- onClick={() => setKeepHomingLogsOpen(true)}
- className="gap-2"
- >
- <Eye className="h-4 w-4" />
- Keep Open
- </Button>
- <Button
- onClick={() => {
- setHomingJustCompleted(false)
- setKeepHomingLogsOpen(false)
- }}
- className="gap-2"
- >
- <X className="h-4 w-4" />
- Dismiss
- </Button>
- </>
- ) : (
- <Button
- onClick={() => {
- setHomingJustCompleted(false)
- setKeepHomingLogsOpen(false)
- }}
- className="gap-2"
- >
- <X className="h-4 w-4" />
- Close Logs
- </Button>
- )}
- </div>
- )}
- {/* Dismiss button during homing */}
- {isHoming && !homingJustCompleted && (
- <div className="flex justify-center">
- <Button
- variant="ghost"
- onClick={() => setHomingDismissed(true)}
- className="gap-2 text-muted-foreground"
- >
- <EyeOff className="h-4 w-4" />
- Dismiss
- </Button>
- </div>
- )}
- {/* Hint */}
- {!homingJustCompleted && (
- <p className="text-center text-xs text-muted-foreground">
- {isHoming
- ? 'Homing will continue in the background'
- : 'Make sure the backend server is running on port 8080'
- }
- </p>
- )}
- </div>
- </div>
- )}
- {/* Header - Floating Pill */}
- <header className="fixed top-0 left-0 right-0 z-40 pt-safe">
- {/* Blurry backdrop behind header - only on Browse page where content scrolls under */}
- {location.pathname === '/' && (
- <div className="absolute inset-0 bg-background/80 backdrop-blur-md supports-[backdrop-filter]:bg-background/50" style={{ height: 'calc(5rem + env(safe-area-inset-top, 0px))' }} />
- )}
- <div className="relative w-full max-w-5xl mx-auto px-3 sm:px-4 pt-3 pointer-events-none">
- <div className="flex h-12 items-center justify-between px-4 rounded-full bg-card shadow-lg border border-border pointer-events-auto">
- <div className="flex items-center gap-2">
- <Link to="/">
- <img
- src={customLogo ? apiClient.getAssetUrl(`/static/custom/${customLogo}`) : apiClient.getAssetUrl('/static/android-chrome-192x192.png')}
- alt={displayName}
- className="w-8 h-8 rounded-full object-cover"
- />
- </Link>
- <TableSelector>
- <button className="flex items-center gap-1.5 hover:opacity-80 transition-opacity group">
- <ShinyText
- text={displayName}
- className="font-semibold text-lg"
- speed={4}
- color={isDark ? '#a8a8a8' : '#555555'}
- shineColor={isDark ? '#ffffff' : '#999999'}
- spread={75}
- />
- <ChevronDown className="h-3.5 w-3.5 text-muted-foreground group-hover:text-foreground transition-colors" />
- <span
- className={`w-2 h-2 rounded-full ${
- !isBackendConnected
- ? 'bg-gray-400'
- : isConnected
- ? 'bg-green-500 animate-pulse'
- : 'bg-red-500'
- }`}
- title={
- !isBackendConnected
- ? 'Backend not connected'
- : isConnected
- ? 'Table connected'
- : 'Table disconnected'
- }
- />
- </button>
- </TableSelector>
- </div>
- {/* Desktop actions */}
- <div className="hidden md:flex items-center gap-0 ml-2">
- <Popover>
- <PopoverTrigger asChild>
- <Button
- variant="ghost"
- size="icon"
- className="rounded-full"
- aria-label="Open menu"
- >
- <Menu className="h-5 w-5" />
- </Button>
- </PopoverTrigger>
- <PopoverContent align="end" className="w-56 p-2">
- <div className="flex flex-col gap-1">
- <button
- onClick={() => setIsDark(!isDark)}
- className="flex items-center gap-3 w-full px-3 py-2 text-sm rounded-md hover:bg-accent transition-colors"
- >
- {isDark ? <Sun className="h-5 w-5" /> : <Moon className="h-5 w-5" />}
- {isDark ? 'Light Mode' : 'Dark Mode'}
- </button>
- <button
- onClick={handleToggleLogs}
- className="flex items-center gap-3 w-full px-3 py-2 text-sm rounded-md hover:bg-accent transition-colors"
- >
- <FileText className="h-5 w-5" />
- View Logs
- </button>
- <Separator className="my-1" />
- <button
- onClick={handleRestart}
- className="flex items-center gap-3 w-full px-3 py-2 text-sm rounded-md hover:bg-accent transition-colors text-amber-500"
- >
- <RotateCcw className="h-5 w-5" />
- Restart Docker
- </button>
- <button
- onClick={handleShutdown}
- className="flex items-center gap-3 w-full px-3 py-2 text-sm rounded-md hover:bg-accent transition-colors text-red-500"
- >
- <Power className="h-5 w-5" />
- Shutdown
- </button>
- </div>
- </PopoverContent>
- </Popover>
- </div>
- {/* Mobile actions */}
- <div className="flex md:hidden items-center gap-0 ml-2">
- <Popover open={isMobileMenuOpen} onOpenChange={setIsMobileMenuOpen}>
- <PopoverTrigger asChild>
- <Button
- variant="ghost"
- size="icon"
- className="rounded-full"
- aria-label="Open menu"
- >
- {isMobileMenuOpen ? <X className="h-5 w-5" /> : <Menu className="h-5 w-5" />}
- </Button>
- </PopoverTrigger>
- <PopoverContent align="end" className="w-56 p-2">
- <div className="flex flex-col gap-1">
- <button
- onClick={() => {
- setIsDark(!isDark)
- setIsMobileMenuOpen(false)
- }}
- className="flex items-center gap-3 w-full px-3 py-2 text-sm rounded-md hover:bg-accent transition-colors"
- >
- {isDark ? <Sun className="h-5 w-5" /> : <Moon className="h-5 w-5" />}
- {isDark ? 'Light Mode' : 'Dark Mode'}
- </button>
- <button
- onClick={() => {
- handleToggleLogs()
- setIsMobileMenuOpen(false)
- }}
- className="flex items-center gap-3 w-full px-3 py-2 text-sm rounded-md hover:bg-accent transition-colors"
- >
- <FileText className="h-5 w-5" />
- View Logs
- </button>
- <Separator className="my-1" />
- <button
- onClick={() => {
- handleRestart()
- setIsMobileMenuOpen(false)
- }}
- className="flex items-center gap-3 w-full px-3 py-2 text-sm rounded-md hover:bg-accent transition-colors text-amber-500"
- >
- <RotateCcw className="h-5 w-5" />
- Restart Docker
- </button>
- <button
- onClick={() => {
- handleShutdown()
- setIsMobileMenuOpen(false)
- }}
- className="flex items-center gap-3 w-full px-3 py-2 text-sm rounded-md hover:bg-accent transition-colors text-red-500"
- >
- <Power className="h-5 w-5" />
- Shutdown
- </button>
- </div>
- </PopoverContent>
- </Popover>
- </div>
- </div>
- </div>
- </header>
- {/* Main Content */}
- <main
- className={`container mx-auto px-4 transition-all duration-300 ${
- !isLogsOpen && !isNowPlayingOpen ? 'pb-20' :
- !isLogsOpen && isNowPlayingOpen ? 'pb-80' : ''
- }`}
- style={{
- paddingTop: 'calc(4.5rem + env(safe-area-inset-top, 0px))',
- paddingBottom: isLogsOpen
- ? isNowPlayingOpen
- ? logsDrawerHeight + 256 + 64 // drawer + now playing + nav
- : logsDrawerHeight + 64 // drawer + nav
- : undefined
- }}
- >
- <Outlet />
- </main>
- {/* Now Playing Bar */}
- <NowPlayingBar
- isLogsOpen={isLogsOpen}
- logsDrawerHeight={logsDrawerHeight}
- isVisible={isNowPlayingOpen}
- openExpanded={openNowPlayingExpanded}
- onClose={() => setIsNowPlayingOpen(false)}
- />
- {/* Logs Drawer */}
- <div
- className={`fixed left-0 right-0 z-30 bg-background border-t border-border ${
- isResizing ? '' : 'transition-[height] duration-300'
- }`}
- style={{
- height: isLogsOpen ? logsDrawerHeight : 0,
- bottom: 'calc(4rem + env(safe-area-inset-bottom, 0px))'
- }}
- >
- {isLogsOpen && (
- <>
- {/* Resize Handle */}
- <div
- className="absolute top-0 left-0 right-0 h-2 cursor-ns-resize flex items-center justify-center group hover:bg-primary/10 -translate-y-1/2 z-10"
- onMouseDown={handleResizeStart}
- onTouchStart={handleResizeStart}
- >
- <div className="w-12 h-1 rounded-full bg-border group-hover:bg-primary transition-colors" />
- </div>
- {/* Logs Header */}
- <div className="flex items-center justify-between px-4 py-2 border-b bg-muted/50">
- <div className="flex items-center gap-3">
- <span className="text-sm font-medium">Application Logs</span>
- <select
- value={logLevelFilter}
- onChange={(e) => setLogLevelFilter(e.target.value)}
- className="text-xs bg-background border rounded px-2 py-1"
- >
- <option value="ALL">All Levels</option>
- <option value="DEBUG">Debug</option>
- <option value="INFO">Info</option>
- <option value="WARNING">Warning</option>
- <option value="ERROR">Error</option>
- </select>
- <span className="text-xs text-muted-foreground">
- {filteredLogs.length}{logsTotal > 0 ? ` of ${logsTotal}` : ''} entries
- {logsHasMore && <span className="text-primary ml-1">↑ scroll for more</span>}
- </span>
- </div>
- <div className="flex items-center gap-1">
- <Button
- variant="ghost"
- size="icon-sm"
- onClick={handleCopyLogs}
- className="rounded-full"
- title="Copy logs"
- >
- <Copy className="h-4 w-4" />
- </Button>
- <Button
- variant="ghost"
- size="icon-sm"
- onClick={handleDownloadLogs}
- className="rounded-full"
- title="Download logs"
- >
- <Download className="h-4 w-4" />
- </Button>
- <Button
- variant="ghost"
- size="icon-sm"
- onClick={() => setIsLogsOpen(false)}
- className="rounded-full"
- title="Close"
- >
- <X className="h-4 w-4" />
- </Button>
- </div>
- </div>
- {/* Logs Content */}
- <div
- ref={logsContainerRef}
- className="h-[calc(100%-40px)] overflow-auto overscroll-contain p-3 font-mono text-xs space-y-0.5"
- >
- {/* Loading indicator for older logs */}
- {isLoadingMoreLogs && (
- <div className="flex items-center justify-center gap-2 py-2 text-muted-foreground">
- <Loader2 className="h-3.5 w-3.5 animate-spin" />
- <span>Loading older logs...</span>
- </div>
- )}
- {/* Load more hint */}
- {logsHasMore && !isLoadingMoreLogs && (
- <div className="text-center py-2 text-muted-foreground text-xs">
- ↑ Scroll up to load older logs
- </div>
- )}
- {filteredLogs.length > 0 ? (
- filteredLogs.map((log, i) => (
- <div key={i} className="py-0.5 flex gap-2">
- <span className="text-muted-foreground shrink-0">
- {formatTimestamp(log.timestamp)}
- </span>
- <span className={`shrink-0 font-semibold ${
- log.level === 'ERROR' ? 'text-red-500' :
- log.level === 'WARNING' ? 'text-amber-500' :
- log.level === 'DEBUG' ? 'text-muted-foreground' :
- 'text-foreground'
- }`}>
- [{log.level || 'LOG'}]
- </span>
- <span className="break-all">{log.message || ''}</span>
- </div>
- ))
- ) : (
- <p className="text-muted-foreground text-center py-4">No logs available</p>
- )}
- </div>
- </>
- )}
- </div>
- {/* Floating Now Playing Button - hidden when Now Playing bar is open */}
- {!isNowPlayingOpen && (
- <button
- onClick={() => setIsNowPlayingOpen(true)}
- className="fixed z-40 left-1/2 -translate-x-1/2 flex items-center gap-2 px-4 py-2 rounded-full bg-card border border-border shadow-lg transition-all hover:shadow-xl hover:scale-105 active:scale-95"
- style={{ bottom: 'calc(4.5rem + env(safe-area-inset-bottom, 0px))' }}
- aria-label={isCurrentlyPlaying ? 'Now Playing' : 'Not Playing'}
- >
- {isCurrentlyPlaying ? (
- <PlayCircle className="h-5 w-5 text-primary" />
- ) : (
- <StopCircle className="h-5 w-5 text-muted-foreground" />
- )}
- <span className="text-sm font-medium">
- {isCurrentlyPlaying ? 'Now Playing' : 'Not Playing'}
- </span>
- </button>
- )}
- {/* Bottom Navigation */}
- <nav className="fixed bottom-0 left-0 right-0 z-40 border-t border-border bg-card pb-safe">
- <div className="max-w-5xl mx-auto grid grid-cols-5 h-16">
- {navItems.map((item) => {
- const isActive = location.pathname === item.path
- return (
- <Link
- key={item.path}
- to={item.path}
- className={`relative flex flex-col items-center justify-center gap-1 transition-all duration-200 ${
- isActive
- ? 'text-primary'
- : 'text-muted-foreground hover:text-foreground active:scale-95'
- }`}
- >
- {/* Active indicator pill */}
- {isActive && (
- <span className="absolute -top-0.5 w-8 h-1 rounded-full bg-primary" />
- )}
- <item.icon className="h-5 w-5" strokeWidth={isActive ? 2.5 : 1.5} />
- <span className="text-xs font-medium">{item.label}</span>
- </Link>
- )
- })}
- </div>
- </nav>
- </div>
- )
- }
|