mirror of
https://github.com/mauriceboe/TREK.git
synced 2026-06-20 13:51:45 +00:00
9614a5cf84
Replace the static, color-only inline `style={{ ... 'var(--bg-primary)' ... }}` props with the new semantic Tailwind utilities (bg-surface, text-content, border-edge, ...) wherever the result is byte-identical; dynamic/conditional theme styles and hardcoded status colors are left inline. Extract the Atlas country-search autocomplete, the Admin update banner, and two Journey dialogs into their own presentational components to shrink the oversized page files, keeping behaviour and markup identical.
301 lines
16 KiB
TypeScript
301 lines
16 KiB
TypeScript
import React, { useState, useEffect, useRef } from 'react'
|
|
import ReactDOM from 'react-dom'
|
|
import { Link, useNavigate, useLocation } from 'react-router-dom'
|
|
import { useAuthStore } from '../../store/authStore'
|
|
import { useSettingsStore } from '../../store/settingsStore'
|
|
import { useAddonStore } from '../../store/addonStore'
|
|
import { useTranslation } from '../../i18n'
|
|
import { Plane, LogOut, Settings, ChevronDown, Shield, ArrowLeft, Users, Moon, Sun, Monitor, CalendarDays, Briefcase, Globe, Compass } from 'lucide-react'
|
|
import type { LucideIcon } from 'lucide-react'
|
|
import InAppNotificationBell from './InAppNotificationBell.tsx'
|
|
|
|
const ADDON_ICONS: Record<string, LucideIcon> = { CalendarDays, Briefcase, Globe, Compass }
|
|
|
|
interface NavbarProps {
|
|
tripTitle?: string
|
|
tripId?: string
|
|
onBack?: () => void
|
|
showBack?: boolean
|
|
onShare?: () => void
|
|
}
|
|
|
|
interface Addon {
|
|
id: string
|
|
name: string
|
|
icon: string
|
|
type: string
|
|
enabled: boolean
|
|
}
|
|
|
|
export default function Navbar({ tripTitle, tripId, onBack, showBack, onShare }: NavbarProps): React.ReactElement {
|
|
const { user, logout, isPrerelease, appVersion } = useAuthStore()
|
|
const { settings, updateSetting } = useSettingsStore()
|
|
const { addons: allAddons, loadAddons } = useAddonStore()
|
|
const { t, locale } = useTranslation()
|
|
const navigate = useNavigate()
|
|
const location = useLocation()
|
|
const [userMenuOpen, setUserMenuOpen] = useState<boolean>(false)
|
|
const [scrolled, setScrolled] = useState<boolean>(false)
|
|
const darkMode = settings.dark_mode
|
|
const dark = darkMode === true || darkMode === 'dark' || (darkMode === 'auto' && window.matchMedia('(prefers-color-scheme: dark)').matches)
|
|
|
|
useEffect(() => {
|
|
const onScroll = () => setScrolled(window.scrollY > 8 || (document.body.scrollTop || 0) > 8)
|
|
onScroll()
|
|
window.addEventListener('scroll', onScroll, { passive: true })
|
|
document.body.addEventListener('scroll', onScroll, { passive: true })
|
|
return () => {
|
|
window.removeEventListener('scroll', onScroll)
|
|
document.body.removeEventListener('scroll', onScroll)
|
|
}
|
|
}, [])
|
|
|
|
// Only show 'global' type addons in the navbar — 'integration' addons have no dedicated page
|
|
const globalAddons = allAddons.filter((a: Addon) => a.type === 'global' && a.enabled)
|
|
|
|
useEffect(() => {
|
|
if (user) loadAddons()
|
|
}, [user, location.pathname])
|
|
|
|
const handleLogout = () => {
|
|
logout()
|
|
navigate('/login', { state: { noRedirect: true } })
|
|
}
|
|
|
|
// Keep track of the pending theme-transition cleanup so we can cancel it
|
|
// on unmount. Without this the timer fires after jsdom teardown in unit
|
|
// tests (document is gone) and triggers an unhandled ReferenceError that
|
|
// trips vitest's exit code.
|
|
const themeTransitionTimer = useRef<number | null>(null)
|
|
useEffect(() => () => {
|
|
if (themeTransitionTimer.current !== null) {
|
|
window.clearTimeout(themeTransitionTimer.current)
|
|
themeTransitionTimer.current = null
|
|
}
|
|
}, [])
|
|
|
|
const toggleDarkMode = () => {
|
|
document.documentElement.classList.add('trek-theme-transitioning')
|
|
updateSetting('dark_mode', dark ? 'light' : 'dark').catch(() => {})
|
|
if (themeTransitionTimer.current !== null) window.clearTimeout(themeTransitionTimer.current)
|
|
themeTransitionTimer.current = window.setTimeout(() => {
|
|
document.documentElement.classList.remove('trek-theme-transitioning')
|
|
themeTransitionTimer.current = null
|
|
}, 360)
|
|
}
|
|
|
|
const getAddonName = (addon: Addon): string => {
|
|
const key = `admin.addons.catalog.${addon.id}.name`
|
|
const translated = t(key)
|
|
return translated !== key ? translated : addon.name
|
|
}
|
|
|
|
return (
|
|
<nav style={{
|
|
background: dark
|
|
? (scrolled ? 'rgba(9,9,11,0.78)' : 'rgba(9,9,11,0.95)')
|
|
: (scrolled ? 'rgba(255,255,255,0.72)' : 'rgba(255,255,255,0.95)'),
|
|
backdropFilter: scrolled ? 'blur(28px) saturate(180%)' : 'blur(20px)',
|
|
WebkitBackdropFilter: scrolled ? 'blur(28px) saturate(180%)' : 'blur(20px)',
|
|
borderBottom: `1px solid ${dark ? 'rgba(255,255,255,0.07)' : 'rgba(0,0,0,0.07)'}`,
|
|
boxShadow: scrolled
|
|
? (dark ? '0 4px 24px rgba(0,0,0,0.35)' : '0 4px 24px rgba(0,0,0,0.08)')
|
|
: (dark ? '0 1px 12px rgba(0,0,0,0.2)' : '0 1px 12px rgba(0,0,0,0.05)'),
|
|
touchAction: 'manipulation',
|
|
paddingTop: 'env(safe-area-inset-top, 0px)',
|
|
height: 'var(--nav-h)',
|
|
transition: 'background 240ms cubic-bezier(0.23,1,0.32,1), backdrop-filter 240ms cubic-bezier(0.23,1,0.32,1), box-shadow 240ms cubic-bezier(0.23,1,0.32,1)',
|
|
}} className="hidden md:flex items-center px-4 gap-4 fixed top-0 left-0 right-0 z-[200]">
|
|
{/* Left side */}
|
|
<div className="flex items-center gap-3 min-w-0">
|
|
{showBack && (
|
|
<button onClick={onBack}
|
|
className="trek-back-btn p-1.5 rounded-lg transition-colors flex items-center gap-1.5 text-sm flex-shrink-0 text-content-muted"
|
|
onMouseEnter={e => e.currentTarget.style.background = 'var(--bg-hover)'}
|
|
onMouseLeave={e => e.currentTarget.style.background = 'transparent'}>
|
|
<ArrowLeft className="trek-back-icon w-4 h-4" />
|
|
<span className="hidden sm:inline">{t('common.back')}</span>
|
|
</button>
|
|
)}
|
|
|
|
<Link to="/dashboard" className="flex items-center transition-colors flex-shrink-0">
|
|
<img src={dark ? '/icons/icon-white.svg' : '/icons/icon-dark.svg'} alt="TREK" className="sm:hidden" style={{ height: 22, width: 22 }} />
|
|
<img src={dark ? '/logo-light.svg' : '/logo-dark.svg'} alt="TREK" className="hidden sm:block" style={{ height: 28 }} />
|
|
</Link>
|
|
|
|
{tripTitle && (
|
|
<>
|
|
<span className="hidden sm:inline text-content-faint">/</span>
|
|
<span className="hidden sm:inline text-sm font-medium truncate max-w-48 text-content-muted">
|
|
{tripTitle}
|
|
</span>
|
|
</>
|
|
)}
|
|
</div>
|
|
|
|
{/* Centred liquid-glass tab menu (design handoff). Absolutely positioned so
|
|
the left brand block and the right action cluster keep their layout. */}
|
|
{globalAddons.length > 0 && !tripTitle && (
|
|
<div
|
|
className="trek-nav-pill"
|
|
style={{
|
|
position: 'absolute', left: '50%', top: '50%', transform: 'translate(-50%, -50%)',
|
|
display: 'flex', gap: 4, padding: 4, borderRadius: 14,
|
|
background: dark ? 'rgba(255,255,255,0.06)' : 'rgba(0,0,0,0.04)',
|
|
backdropFilter: 'blur(20px) saturate(180%)', WebkitBackdropFilter: 'blur(20px) saturate(180%)',
|
|
border: `1px solid ${dark ? 'rgba(255,255,255,0.08)' : 'rgba(0,0,0,0.05)'}`,
|
|
}}
|
|
>
|
|
{[{ id: '__trips', path: '/dashboard', label: t('nav.myTrips'), Icon: Briefcase },
|
|
...globalAddons.map(a => ({ id: a.id, path: `/${a.id}`, label: getAddonName(a), Icon: ADDON_ICONS[a.icon] || CalendarDays }))
|
|
].map(tab => {
|
|
const isActive = location.pathname === tab.path
|
|
return (
|
|
<Link key={tab.id} to={tab.path}
|
|
className="flex items-center gap-1.5 transition-colors"
|
|
style={{
|
|
padding: '5px 16px', borderRadius: 9, fontSize: 13.5, fontWeight: 500,
|
|
color: isActive ? 'var(--text-primary)' : 'var(--text-muted)',
|
|
background: isActive ? 'var(--bg-card)' : 'transparent',
|
|
boxShadow: isActive ? '0 1px 2px rgba(0,0,0,0.06), 0 2px 6px rgba(0,0,0,0.05)' : 'none',
|
|
}}
|
|
onMouseEnter={e => { if (!isActive) e.currentTarget.style.color = 'var(--text-primary)' }}
|
|
onMouseLeave={e => { if (!isActive) e.currentTarget.style.color = 'var(--text-muted)' }}>
|
|
<tab.Icon className="w-4 h-4" />
|
|
<span>{tab.label}</span>
|
|
</Link>
|
|
)
|
|
})}
|
|
</div>
|
|
)}
|
|
|
|
{/* Spacer */}
|
|
<div className="flex-1" />
|
|
|
|
{/* Share button */}
|
|
{onShare && (
|
|
<button onClick={onShare}
|
|
className="flex items-center gap-1.5 py-1.5 px-3 rounded-lg border transition-colors text-sm font-medium flex-shrink-0 border-edge text-content-secondary"
|
|
style={{ background: 'var(--bg-card)' }}
|
|
onMouseEnter={e => e.currentTarget.style.background = 'var(--bg-hover)'}
|
|
onMouseLeave={e => e.currentTarget.style.background = 'var(--bg-card)'}>
|
|
<Users className="w-4 h-4" />
|
|
<span className="hidden sm:inline">{t('nav.share')}</span>
|
|
</button>
|
|
)}
|
|
|
|
{/* Prerelease badge */}
|
|
{isPrerelease && appVersion && (
|
|
<span
|
|
className="hidden sm:flex items-center gap-1.5 px-2.5 py-1 rounded-full text-[11px] font-semibold flex-shrink-0"
|
|
style={{ background: 'rgba(245,158,11,0.15)', color: '#d97706', border: '1px solid rgba(245,158,11,0.3)' }}
|
|
>
|
|
<span className="w-1.5 h-1.5 rounded-full flex-shrink-0" style={{ background: '#f59e0b' }} />
|
|
{appVersion}
|
|
</span>
|
|
)}
|
|
|
|
{/* Dark mode toggle (light ↔ dark, overrides auto) — hidden on mobile */}
|
|
<button onClick={toggleDarkMode} title={dark ? t('nav.lightMode') : t('nav.darkMode')}
|
|
className="p-2 rounded-lg transition-colors flex-shrink-0 hidden sm:flex relative w-8 h-8 items-center justify-center text-content-muted"
|
|
onMouseEnter={e => e.currentTarget.style.background = 'var(--bg-hover)'}
|
|
onMouseLeave={e => e.currentTarget.style.background = 'transparent'}>
|
|
<Sun className="w-4 h-4 absolute transition-[transform,opacity] duration-300 ease-[cubic-bezier(0.23,1,0.32,1)]"
|
|
style={{ opacity: dark ? 1 : 0, transform: dark ? 'rotate(0deg) scale(1)' : 'rotate(-90deg) scale(0.6)' }} />
|
|
<Moon className="w-4 h-4 absolute transition-[transform,opacity] duration-300 ease-[cubic-bezier(0.23,1,0.32,1)]"
|
|
style={{ opacity: dark ? 0 : 1, transform: dark ? 'rotate(90deg) scale(0.6)' : 'rotate(0deg) scale(1)' }} />
|
|
</button>
|
|
|
|
{/* Notification bell — only in trip view on mobile, everywhere on desktop */}
|
|
{user && tripId && <InAppNotificationBell />}
|
|
{user && !tripId && <span className="hidden sm:block"><InAppNotificationBell /></span>}
|
|
|
|
{/* User menu */}
|
|
{user && (
|
|
<div className="relative">
|
|
<button onClick={() => setUserMenuOpen(!userMenuOpen)}
|
|
className="flex items-center gap-2 py-1.5 px-3 rounded-lg transition-colors"
|
|
onMouseEnter={e => e.currentTarget.style.background = 'var(--bg-hover)'}
|
|
onMouseLeave={e => e.currentTarget.style.background = 'transparent'}>
|
|
{user.avatar_url ? (
|
|
<img src={user.avatar_url} alt="" style={{ width: 28, height: 28, borderRadius: '50%', objectFit: 'cover' }} />
|
|
) : (
|
|
<div className="w-7 h-7 rounded-full flex items-center justify-center text-xs font-bold"
|
|
style={{ background: dark ? '#e2e8f0' : '#111827', color: dark ? '#0f172a' : '#ffffff' }}>
|
|
{user.username?.charAt(0).toUpperCase()}
|
|
</div>
|
|
)}
|
|
<span className="text-sm hidden sm:inline max-w-24 truncate text-content-secondary">
|
|
{user.username}
|
|
</span>
|
|
<ChevronDown className="w-4 h-4 text-content-faint" />
|
|
</button>
|
|
|
|
{userMenuOpen && ReactDOM.createPortal(
|
|
<>
|
|
<div style={{ position: 'fixed', inset: 0, zIndex: 9998 }} onClick={() => setUserMenuOpen(false)} />
|
|
<div className="trek-menu-enter w-52 rounded-xl shadow-xl border overflow-hidden bg-surface-card border-edge" style={{ position: 'fixed', top: 'var(--nav-h)', right: 8, zIndex: 9999 }}>
|
|
<div className="px-4 py-3 border-b border-edge-secondary">
|
|
<p className="text-sm font-medium text-content">{user.username}</p>
|
|
<p className="text-xs truncate text-content-muted">{user.email}</p>
|
|
{user.role === 'admin' && (
|
|
<span className="inline-flex items-center gap-1 text-xs font-medium mt-1 text-content-secondary">
|
|
<Shield className="w-3 h-3" /> {t('nav.administrator')}
|
|
</span>
|
|
)}
|
|
</div>
|
|
|
|
<div className="py-1">
|
|
<Link to="/settings" onClick={() => setUserMenuOpen(false)}
|
|
className="flex items-center gap-2 px-4 py-2 text-sm transition-colors text-content-secondary"
|
|
onMouseEnter={e => e.currentTarget.style.background = 'var(--bg-hover)'}
|
|
onMouseLeave={e => e.currentTarget.style.background = 'transparent'}>
|
|
<Settings className="w-4 h-4" />
|
|
{t('nav.settings')}
|
|
</Link>
|
|
|
|
{user.role === 'admin' && (
|
|
<Link to="/admin" onClick={() => setUserMenuOpen(false)}
|
|
className="flex items-center gap-2 px-4 py-2 text-sm transition-colors text-content-secondary"
|
|
onMouseEnter={e => e.currentTarget.style.background = 'var(--bg-hover)'}
|
|
onMouseLeave={e => e.currentTarget.style.background = 'transparent'}>
|
|
<Shield className="w-4 h-4" />
|
|
{t('nav.admin')}
|
|
</Link>
|
|
)}
|
|
</div>
|
|
|
|
<div className="py-1 border-t border-edge-secondary">
|
|
<button onClick={handleLogout}
|
|
className="flex items-center gap-2 w-full px-4 py-2 text-sm text-red-500 hover:bg-red-500/10 transition-colors">
|
|
<LogOut className="w-4 h-4" />
|
|
{t('nav.logout')}
|
|
</button>
|
|
{appVersion && (
|
|
<div className="px-4 pt-2 pb-2.5 text-center border-t border-edge-secondary" style={{ marginTop: 4 }}>
|
|
<div style={{ display: 'flex', alignItems: 'center', justifyContent: 'center', gap: 6 }}>
|
|
<div style={{ display: 'inline-flex', alignItems: 'center', gap: 5, background: 'var(--bg-tertiary)', borderRadius: 99, padding: '4px 12px' }}>
|
|
<img src={dark ? '/text-light.svg' : '/text-dark.svg'} alt="TREK" style={{ height: 10, opacity: 0.5 }} />
|
|
<span style={{ fontSize: 10, fontWeight: 600, color: 'var(--text-faint)' }}>v{appVersion}</span>
|
|
</div>
|
|
<a href="https://discord.gg/NhZBDSd4qW" target="_blank" rel="noopener noreferrer"
|
|
style={{ display: 'inline-flex', alignItems: 'center', justifyContent: 'center', width: 24, height: 24, borderRadius: 99, background: 'var(--bg-tertiary)', transition: 'background 0.15s' }}
|
|
onMouseEnter={e => e.currentTarget.style.background = '#5865F220'}
|
|
onMouseLeave={e => e.currentTarget.style.background = 'var(--bg-tertiary)'}
|
|
title="Discord">
|
|
<svg width="12" height="12" viewBox="0 0 24 24" fill="var(--text-faint)"><path d="M20.317 4.37a19.791 19.791 0 0 0-4.885-1.515.074.074 0 0 0-.079.037c-.21.375-.444.864-.608 1.25a18.27 18.27 0 0 0-5.487 0 12.64 12.64 0 0 0-.617-1.25.077.077 0 0 0-.079-.037A19.736 19.736 0 0 0 3.677 4.37a.07.07 0 0 0-.032.027C.533 9.046-.32 13.58.099 18.057a.082.082 0 0 0 .031.057 19.9 19.9 0 0 0 5.993 3.03.078.078 0 0 0 .084-.028c.462-.63.874-1.295 1.226-1.994a.076.076 0 0 0-.041-.106 13.107 13.107 0 0 1-1.872-.892.077.077 0 0 1-.008-.128 10.2 10.2 0 0 0 .372-.292.074.074 0 0 1 .077-.01c3.928 1.793 8.18 1.793 12.062 0a.074.074 0 0 1 .078.01c.12.098.246.198.373.292a.077.077 0 0 1-.006.127 12.299 12.299 0 0 1-1.873.892.077.077 0 0 0-.041.107c.36.698.772 1.362 1.225 1.993a.076.076 0 0 0 .084.028 19.839 19.839 0 0 0 6.002-3.03.077.077 0 0 0 .032-.054c.5-5.177-.838-9.674-3.549-13.66a.061.061 0 0 0-.031-.03zM8.02 15.33c-1.183 0-2.157-1.085-2.157-2.419 0-1.333.956-2.419 2.157-2.419 1.21 0 2.176 1.095 2.157 2.42 0 1.333-.956 2.418-2.157 2.418zm7.975 0c-1.183 0-2.157-1.085-2.157-2.419 0-1.333.955-2.419 2.157-2.419 1.21 0 2.176 1.095 2.157 2.42 0 1.333-.946 2.418-2.157 2.418z"/></svg>
|
|
</a>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</>,
|
|
document.body
|
|
)}
|
|
</div>
|
|
)}
|
|
</nav>
|
|
)
|
|
}
|