| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786 |
- import { useState, useCallback, useRef, useEffect } from 'react';
- import { useQuery } from '@tanstack/react-query';
- import { useOutletContext } from 'react-router-dom';
- import { useTranslation } from 'react-i18next';
- import type { SpoolBuddyOutletContext } from '../../components/spoolbuddy/SpoolBuddyLayout';
- import { spoolbuddyApi, type SpoolBuddyDevice, type DaemonUpdateCheck } from '../../api/client';
- function formatUptime(seconds: number): string {
- if (seconds < 60) return `${seconds}s`;
- if (seconds < 3600) return `${Math.floor(seconds / 60)}m`;
- const h = Math.floor(seconds / 3600);
- const m = Math.floor((seconds % 3600) / 60);
- return `${h}h ${m}m`;
- }
- function formatDateTime(iso: string | null): string {
- if (!iso) return '-';
- try {
- const d = new Date(iso);
- return d.toLocaleDateString(undefined, { month: 'short', day: 'numeric' }) + ' ' +
- d.toLocaleTimeString(undefined, { hour: '2-digit', minute: '2-digit' });
- } catch {
- return '-';
- }
- }
- const BLANK_OPTIONS = [
- { label: 'Off', value: 0 },
- { label: '1m', value: 60 },
- { label: '2m', value: 120 },
- { label: '5m', value: 300 },
- { label: '10m', value: 600 },
- { label: '30m', value: 1800 },
- ];
- // --- Device Tab ---
- function DeviceTab({ device }: { device: SpoolBuddyDevice }) {
- const { t } = useTranslation();
- return (
- <div className="space-y-4">
- {/* About */}
- <div className="bg-zinc-800 rounded-lg p-4">
- <div className="flex items-center gap-3 mb-2">
- <img src="/img/spoolbuddy_logo_dark_small.png" alt="SpoolBuddy" className="h-7 w-auto" />
- </div>
- <p className="text-xs text-zinc-500 mb-1">Part of Bambuddy</p>
- <span className="text-xs text-zinc-500">github.com/maziggy/bambuddy</span>
- </div>
- {/* NFC Reader + Device Info side by side */}
- <div className="grid grid-cols-2 gap-3">
- {/* NFC Reader */}
- <div className="bg-zinc-800 rounded-lg p-3">
- <h3 className="text-sm font-semibold text-zinc-300 mb-2">
- {t('spoolbuddy.settings.nfcReader', 'NFC Reader')}
- </h3>
- <div className="space-y-1.5 text-xs">
- <div className="flex justify-between">
- <span className="text-zinc-500">{t('spoolbuddy.settings.type', 'Type')}</span>
- <span className="text-zinc-300 font-mono">
- {device.nfc_reader_type || 'N/A'}
- </span>
- </div>
- <div className="flex justify-between">
- <span className="text-zinc-500">{t('spoolbuddy.settings.connection', 'Connection')}</span>
- <span className="text-zinc-300 font-mono">
- {device.nfc_connection || 'N/A'}
- </span>
- </div>
- <div className="flex justify-between items-center">
- <span className="text-zinc-500">{t('spoolbuddy.status.status', 'Status')}</span>
- <div className="flex items-center gap-1.5">
- <div className={`w-2 h-2 rounded-full ${
- device.nfc_ok ? 'bg-green-500' : device.nfc_reader_type ? 'bg-red-500' : 'bg-zinc-600'
- }`} />
- <span className={
- device.nfc_ok ? 'text-green-400' : device.nfc_reader_type ? 'text-red-400' : 'text-zinc-500'
- }>
- {device.nfc_ok
- ? t('spoolbuddy.status.nfcReady', 'Ready')
- : device.nfc_reader_type
- ? t('common.error', 'Error')
- : t('spoolbuddy.settings.notConnected', 'N/A')}
- </span>
- </div>
- </div>
- </div>
- </div>
- {/* Device Info */}
- <div className="bg-zinc-800 rounded-lg p-3">
- <h3 className="text-sm font-semibold text-zinc-300 mb-2">
- {t('spoolbuddy.settings.deviceInfo', 'Device Info')}
- </h3>
- <div className="space-y-1.5 text-xs">
- <div className="flex justify-between">
- <span className="text-zinc-500">{t('spoolbuddy.settings.hostname', 'Host')}</span>
- <span className="text-zinc-300 truncate ml-2">{device.hostname}</span>
- </div>
- <div className="flex justify-between">
- <span className="text-zinc-500">IP</span>
- <span className="text-zinc-300">{device.ip_address}</span>
- </div>
- <div className="flex justify-between">
- <span className="text-zinc-500">{t('spoolbuddy.settings.uptime', 'Uptime')}</span>
- <span className="text-zinc-300">{formatUptime(device.uptime_s)}</span>
- </div>
- <div className="flex justify-between items-center">
- <span className="text-zinc-500">{t('spoolbuddy.status.status', 'Status')}</span>
- <div className="flex items-center gap-1.5">
- <div className={`w-2 h-2 rounded-full ${device.online ? 'bg-green-500' : 'bg-zinc-600'}`} />
- <span className={device.online ? 'text-green-400' : 'text-zinc-500'}>
- {device.online ? t('spoolbuddy.status.online', 'Online') : t('spoolbuddy.status.offline', 'Offline')}
- </span>
- </div>
- </div>
- </div>
- </div>
- </div>
- {/* Device ID (full width, below cards) */}
- <div className="bg-zinc-800 rounded-lg px-3 py-2 flex justify-between items-center text-xs">
- <span className="text-zinc-500">Device ID</span>
- <span className="text-zinc-400 font-mono">{device.device_id}</span>
- </div>
- </div>
- );
- }
- // --- Display Tab ---
- function DisplayTab({ device, onBrightnessChange, onBlankTimeoutChange }: {
- device: SpoolBuddyDevice;
- onBrightnessChange: (value: number) => void;
- onBlankTimeoutChange: (value: number) => void;
- }) {
- const { t } = useTranslation();
- const [brightness, setBrightness] = useState(device.display_brightness);
- const [blankTimeout, setBlankTimeout] = useState(device.display_blank_timeout);
- const [saved, setSaved] = useState(false);
- const debounceRef = useRef<ReturnType<typeof setTimeout>>(undefined);
- const savedTimerRef = useRef<ReturnType<typeof setTimeout>>(undefined);
- // Sync local state when device data updates from server
- useEffect(() => {
- setBrightness(device.display_brightness);
- setBlankTimeout(device.display_blank_timeout);
- }, [device.display_brightness, device.display_blank_timeout]);
- const showSaved = useCallback(() => {
- setSaved(true);
- if (savedTimerRef.current) clearTimeout(savedTimerRef.current);
- savedTimerRef.current = setTimeout(() => setSaved(false), 1500);
- }, []);
- const sendDisplayUpdate = useCallback((b: number, bt: number) => {
- if (debounceRef.current) clearTimeout(debounceRef.current);
- debounceRef.current = setTimeout(() => {
- spoolbuddyApi.updateDisplay(device.device_id, b, bt)
- .then(() => showSaved())
- .catch((e) => console.error('Failed to update display:', e));
- }, 300);
- }, [device.device_id, showSaved]);
- const handleBrightnessChange = (value: number) => {
- setBrightness(value);
- onBrightnessChange(value); // Instant layout update
- sendDisplayUpdate(value, blankTimeout);
- };
- const handleBlankTimeoutChange = (value: number) => {
- setBlankTimeout(value);
- onBlankTimeoutChange(value); // Instant layout update
- sendDisplayUpdate(brightness, value);
- };
- return (
- <div className="space-y-4">
- {/* Brightness */}
- <div className="bg-zinc-800 rounded-lg p-4">
- <div className="flex items-center justify-between mb-3">
- <h3 className="text-sm font-semibold text-zinc-300">
- {t('spoolbuddy.settings.brightness', 'Brightness')}
- </h3>
- {saved && (
- <span className="text-xs text-green-400 flex items-center gap-1 animate-pulse">
- <svg className="w-3 h-3" fill="none" viewBox="0 0 24 24" stroke="currentColor" strokeWidth={3}>
- <path strokeLinecap="round" strokeLinejoin="round" d="M5 13l4 4L19 7" />
- </svg>
- {t('spoolbuddy.settings.saved', 'Saved')}
- </span>
- )}
- </div>
- <div className="flex items-center gap-3">
- <svg className="w-4 h-4 text-zinc-500 flex-shrink-0" fill="none" viewBox="0 0 24 24" stroke="currentColor" strokeWidth={2}>
- <path strokeLinecap="round" strokeLinejoin="round" d="M12 3v1m0 16v1m9-9h-1M4 12H3m15.364 6.364l-.707-.707M6.343 6.343l-.707-.707m12.728 0l-.707.707M6.343 17.657l-.707.707M16 12a4 4 0 11-8 0 4 4 0 018 0z" />
- </svg>
- <input
- type="range"
- min={0}
- max={100}
- value={brightness}
- onChange={(e) => handleBrightnessChange(parseInt(e.target.value))}
- className="flex-1 h-2 bg-zinc-700 rounded-lg appearance-none cursor-pointer accent-green-500"
- />
- <span className="text-sm font-mono text-zinc-400 w-10 text-right">{brightness}%</span>
- </div>
- {!device.has_backlight && (
- <p className="text-xs text-zinc-600 mt-2">
- {t('spoolbuddy.settings.noBacklight', 'No DSI backlight detected. Brightness control requires a DSI display.')}
- </p>
- )}
- </div>
- {/* Screen blank timeout */}
- <div className="bg-zinc-800 rounded-lg p-4">
- <h3 className="text-sm font-semibold text-zinc-300 mb-1">
- {t('spoolbuddy.settings.screenBlank', 'Screen Blank Timeout')}
- </h3>
- <p className="text-xs text-zinc-500 mb-3">
- {t('spoolbuddy.settings.screenBlankDesc', 'Screen turns off after inactivity. Touch to wake.')}
- </p>
- <div className="grid grid-cols-3 gap-2">
- {BLANK_OPTIONS.map((opt) => (
- <button
- key={opt.value}
- onClick={() => handleBlankTimeoutChange(opt.value)}
- className={`px-3 py-2 rounded-lg text-sm font-medium transition-colors min-h-[40px] ${
- blankTimeout === opt.value
- ? 'bg-green-600 text-white'
- : 'bg-zinc-700 text-zinc-300 hover:bg-zinc-600'
- }`}
- >
- {opt.label}
- </button>
- ))}
- </div>
- </div>
- <p className="text-xs text-zinc-600 text-center">
- {t('spoolbuddy.settings.displayNote', 'Brightness is applied as a software filter.')}
- </p>
- </div>
- );
- }
- // --- Scale Tab ---
- function StepIndicator({ step, labels }: { step: 'tare' | 'weight'; labels: { tare: string; weight: string } }) {
- return (
- <div className="flex flex-col items-center w-16 shrink-0 pt-1">
- {/* Step 1 circle */}
- <div className={`flex items-center justify-center w-7 h-7 rounded-full text-xs font-bold ${
- step === 'tare'
- ? 'bg-green-600 text-white'
- : 'bg-green-600/20 text-green-400'
- }`}>
- {step === 'weight' ? (
- <svg className="w-3.5 h-3.5" fill="none" viewBox="0 0 24 24" stroke="currentColor" strokeWidth={3}>
- <path strokeLinecap="round" strokeLinejoin="round" d="M5 13l4 4L19 7" />
- </svg>
- ) : '1'}
- </div>
- <span className={`text-[10px] mt-0.5 ${step === 'tare' ? 'text-green-400 font-medium' : 'text-green-400/60'}`}>
- {labels.tare}
- </span>
- {/* Connector line */}
- <div className={`w-px h-5 my-1 ${step === 'weight' ? 'bg-green-600/40' : 'bg-zinc-700'}`} />
- {/* Step 2 circle */}
- <div className={`flex items-center justify-center w-7 h-7 rounded-full text-xs font-bold ${
- step === 'weight'
- ? 'bg-green-600 text-white'
- : 'bg-zinc-700 text-zinc-500'
- }`}>
- 2
- </div>
- <span className={`text-[10px] mt-0.5 ${step === 'weight' ? 'text-green-400 font-medium' : 'text-zinc-600'}`}>
- {labels.weight}
- </span>
- </div>
- );
- }
- function ScaleTab({ device, weight, weightStable, rawAdc }: {
- device: SpoolBuddyDevice;
- weight: number | null;
- weightStable: boolean;
- rawAdc: number | null;
- }) {
- const { t } = useTranslation();
- const [calStep, setCalStep] = useState<'idle' | 'tare' | 'weight'>('idle');
- const [knownWeight, setKnownWeight] = useState('500');
- const [tareRawAdc, setTareRawAdc] = useState<number | null>(null);
- const [busy, setBusy] = useState(false);
- const [status, setStatus] = useState<{ type: 'ok' | 'error'; msg: string } | null>(null);
- const numpadPress = (key: string) => {
- if (key === 'backspace') {
- setKnownWeight((v) => v.slice(0, -1) || '');
- } else if (key === '.' && !knownWeight.includes('.')) {
- setKnownWeight((v) => v + '.');
- } else if (key >= '0' && key <= '9') {
- setKnownWeight((v) => (v === '0' ? key : v + key));
- }
- };
- const handleTare = async () => {
- setBusy(true);
- setStatus(null);
- try {
- await spoolbuddyApi.tare(device.device_id);
- setStatus({ type: 'ok', msg: t('spoolbuddy.settings.tareSet', 'Tare command sent. Waiting for device...') });
- } catch {
- setStatus({ type: 'error', msg: t('spoolbuddy.settings.tareFailed', 'Failed to send tare command') });
- } finally {
- setBusy(false);
- }
- };
- const handleCalStep = async () => {
- if (calStep === 'tare') {
- setBusy(true);
- setStatus(null);
- try {
- setTareRawAdc(rawAdc);
- await spoolbuddyApi.tare(device.device_id);
- setStatus({ type: 'ok', msg: t('spoolbuddy.settings.zeroSet', 'Zero point set. Place known weight on scale.') });
- setCalStep('weight');
- } catch {
- setStatus({ type: 'error', msg: t('spoolbuddy.settings.tareFailed', 'Failed to send tare command') });
- } finally {
- setBusy(false);
- }
- } else if (calStep === 'weight') {
- const weightNum = parseFloat(knownWeight);
- if (rawAdc === null || !weightNum || weightNum <= 0) return;
- setBusy(true);
- setStatus(null);
- try {
- await spoolbuddyApi.setCalibrationFactor(device.device_id, weightNum, rawAdc, tareRawAdc ?? undefined);
- setStatus({ type: 'ok', msg: t('spoolbuddy.settings.calibrationDone', 'Calibration complete!') });
- setCalStep('idle');
- } catch {
- setStatus({ type: 'error', msg: t('spoolbuddy.settings.calibrationFailed', 'Calibration failed') });
- } finally {
- setBusy(false);
- }
- }
- };
- // --- Idle state: weight card + buttons ---
- if (calStep === 'idle') {
- return (
- <div className="flex flex-col h-full">
- {/* Weight + info card */}
- <div className="bg-zinc-800 rounded-lg p-3 mb-3">
- <div className="flex items-center justify-between">
- <div className="flex items-center gap-2">
- <div className={`w-2 h-2 rounded-full ${weightStable ? 'bg-green-500' : 'bg-amber-500 animate-pulse'}`} />
- <span className="text-lg font-mono text-zinc-200">
- {weight !== null ? `${weight.toFixed(1)} g` : '-- g'}
- </span>
- </div>
- <div className="text-xs text-zinc-500 text-right">
- <span>{t('spoolbuddy.settings.tareOffset', 'Tare')}: {device.tare_offset}</span>
- <span className="mx-1.5">·</span>
- <span>{t('spoolbuddy.settings.calFactor', 'Factor')}: {device.calibration_factor.toFixed(2)}</span>
- </div>
- </div>
- {device.last_calibrated_at && (
- <div className="text-xs text-zinc-600 mt-1">
- {t('spoolbuddy.settings.lastCalibrated', 'Last calibrated')}: {formatDateTime(device.last_calibrated_at)}
- </div>
- )}
- </div>
- {/* Status message */}
- {status && (
- <div className={`rounded-lg px-3 py-2 mb-3 text-sm ${
- status.type === 'ok' ? 'bg-green-900/30 text-green-300 border border-green-800' : 'bg-red-900/30 text-red-300 border border-red-800'
- }`}>
- {status.msg}
- </div>
- )}
- {/* Action buttons */}
- <div className="flex gap-2">
- <button
- onClick={handleTare}
- disabled={busy}
- className="flex-1 px-4 py-2.5 rounded-lg text-sm font-medium bg-zinc-700 text-zinc-200 hover:bg-zinc-600 disabled:opacity-40 transition-colors min-h-[44px] flex items-center justify-center gap-2"
- >
- {busy && (
- <svg className="w-4 h-4 animate-spin" viewBox="0 0 24 24" fill="none">
- <circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4" />
- <path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z" />
- </svg>
- )}
- {t('spoolbuddy.weight.tare', 'Tare')}
- </button>
- <button
- onClick={() => { setCalStep('tare'); setStatus(null); }}
- className="flex-1 px-4 py-2.5 rounded-lg text-sm font-medium bg-green-600 text-white hover:bg-green-700 transition-colors min-h-[44px]"
- >
- {t('spoolbuddy.weight.calibrate', 'Calibrate')}
- </button>
- </div>
- </div>
- );
- }
- // --- Calibration wizard: step indicator left + content right ---
- return (
- <div className="flex gap-3">
- {/* Left: step indicator */}
- <StepIndicator step={calStep} labels={{ tare: t('spoolbuddy.weight.tare', 'Tare'), weight: t('spoolbuddy.settings.knownWeight', 'Known weight') }} />
- {/* Right: content */}
- <div className="flex-1 min-w-0">
- {/* Live weight bar */}
- <div className="flex items-center gap-2 bg-zinc-800 rounded-lg px-3 py-1.5 mb-1.5">
- <div className={`w-2 h-2 rounded-full shrink-0 ${weightStable ? 'bg-green-500' : 'bg-amber-500 animate-pulse'}`} />
- <span className="text-sm font-mono text-zinc-200">
- {weight !== null ? `${weight.toFixed(1)} g` : '-- g'}
- </span>
- <span className={`text-xs ml-auto ${weightStable ? 'text-green-400' : 'text-amber-400'}`}>
- {weightStable ? t('spoolbuddy.settings.stable', 'Stable') : t('spoolbuddy.settings.settling', 'Settling...')}
- </span>
- </div>
- {/* Status message */}
- {status && (
- <div className={`rounded-lg px-3 py-1.5 mb-1.5 text-xs ${
- status.type === 'ok' ? 'bg-green-900/30 text-green-300 border border-green-800' : 'bg-red-900/30 text-red-300 border border-red-800'
- }`}>
- {status.msg}
- </div>
- )}
- {/* Step content */}
- {calStep === 'tare' ? (
- <p className="text-sm text-zinc-300 mb-3">
- {t('spoolbuddy.settings.calStep1', 'Remove all items from the scale and press Set Zero.')}
- </p>
- ) : (
- <>
- <div className="flex items-center gap-2 mb-1.5">
- <span className="text-xs text-zinc-400 shrink-0">{t('spoolbuddy.settings.knownWeight', 'Known weight')}</span>
- <div className="flex-1 bg-zinc-900 border border-zinc-600 rounded px-3 py-1 text-right text-lg font-mono text-zinc-100">
- {knownWeight || '0'}<span className="text-zinc-500 ml-1">g</span>
- </div>
- </div>
- <div className="grid grid-cols-4 gap-1 mb-1.5">
- {['7','8','9','backspace','4','5','6','.','1','2','3','0'].map((key) => (
- <button
- key={key}
- onClick={() => numpadPress(key)}
- className={`rounded text-lg font-medium transition-colors h-[48px] active:scale-95 ${
- key === 'backspace'
- ? 'bg-zinc-700 text-zinc-300 hover:bg-zinc-600'
- : 'bg-zinc-800 text-zinc-100 hover:bg-zinc-700 border border-zinc-700'
- }`}
- >
- {key === 'backspace' ? '\u232B' : key}
- </button>
- ))}
- </div>
- </>
- )}
- {/* Action buttons */}
- <div className="flex gap-2">
- <button
- onClick={() => { setCalStep('idle'); setStatus(null); }}
- className="flex-1 px-4 py-2 rounded-lg text-sm bg-zinc-700 text-zinc-300 hover:bg-zinc-600 transition-colors h-[40px]"
- >
- {t('common.cancel', 'Cancel')}
- </button>
- <button
- onClick={handleCalStep}
- disabled={busy}
- className="flex-1 px-4 py-2 rounded-lg text-sm font-medium bg-green-600 text-white hover:bg-green-700 disabled:opacity-40 transition-colors h-[40px] flex items-center justify-center gap-2"
- >
- {busy && (
- <svg className="w-4 h-4 animate-spin" viewBox="0 0 24 24" fill="none">
- <circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4" />
- <path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z" />
- </svg>
- )}
- {calStep === 'tare' ? t('spoolbuddy.settings.setZero', 'Set Zero') : t('spoolbuddy.settings.calibrateNow', 'Calibrate')}
- </button>
- </div>
- </div>
- </div>
- );
- }
- // --- Updates Tab ---
- function UpdatesTab({ device }: { device: SpoolBuddyDevice }) {
- const { t } = useTranslation();
- const [checking, setChecking] = useState(false);
- const [applying, setApplying] = useState(false);
- const [updateResult, setUpdateResult] = useState<DaemonUpdateCheck | null>(null);
- const [error, setError] = useState<string | null>(null);
- const [includeBeta, setIncludeBeta] = useState(() => {
- try {
- return localStorage.getItem('spoolbuddy-include-beta') === 'true';
- } catch {
- return false;
- }
- });
- const isUpdating = device.update_status === 'pending' || device.update_status === 'updating';
- const toggleBeta = () => {
- const next = !includeBeta;
- setIncludeBeta(next);
- try {
- localStorage.setItem('spoolbuddy-include-beta', String(next));
- } catch {
- // localStorage unavailable
- }
- setUpdateResult(null);
- setError(null);
- };
- const checkForUpdates = async () => {
- setChecking(true);
- setUpdateResult(null);
- setError(null);
- try {
- const result = await spoolbuddyApi.checkDaemonUpdate(device.device_id, includeBeta);
- setUpdateResult(result);
- } catch (e) {
- setError(e instanceof Error ? e.message : 'Failed to check for updates');
- } finally {
- setChecking(false);
- }
- };
- const applyUpdate = async () => {
- setApplying(true);
- setError(null);
- try {
- await spoolbuddyApi.triggerUpdate(device.device_id);
- } catch (e) {
- setError(e instanceof Error ? e.message : 'Failed to trigger update');
- } finally {
- setApplying(false);
- }
- };
- // Show version from device, or from update check result if available
- const displayVersion = device.firmware_version
- || (updateResult?.current_version && updateResult.current_version !== '0.0.0' ? updateResult.current_version : null);
- return (
- <div className="space-y-4">
- {/* Current version */}
- <div className="bg-zinc-800 rounded-lg p-4">
- <h3 className="text-sm font-semibold text-zinc-300 mb-3">
- {t('spoolbuddy.settings.daemonVersion', 'Daemon Version')}
- </h3>
- <div className="flex justify-between items-center text-sm">
- <span className="text-zinc-500">{t('spoolbuddy.settings.currentVersion', 'Current')}</span>
- <span className="text-zinc-200 font-mono">
- {displayVersion || (
- <span className="text-zinc-500 italic">{t('spoolbuddy.settings.versionPending', 'Waiting for daemon...')}</span>
- )}
- </span>
- </div>
- </div>
- {/* Update progress (shown when update is in progress) */}
- {isUpdating && (
- <div className="bg-zinc-800 rounded-lg p-4">
- <div className="flex items-center gap-3">
- <svg className="w-5 h-5 animate-spin text-green-400 flex-shrink-0" viewBox="0 0 24 24" fill="none">
- <circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4" />
- <path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z" />
- </svg>
- <div>
- <p className="text-sm font-medium text-green-300">
- {t('spoolbuddy.settings.updating', 'Updating...')}
- </p>
- <p className="text-xs text-zinc-400 mt-0.5">
- {device.update_message || t('spoolbuddy.settings.updateWaiting', 'Waiting for device...')}
- </p>
- </div>
- </div>
- </div>
- )}
- {/* Update complete */}
- {device.update_status === 'complete' && (
- <div className="rounded-lg p-3 text-sm bg-green-900/30 border border-green-800">
- <div className="flex items-center gap-2">
- <svg className="w-4 h-4 text-green-400 flex-shrink-0" fill="none" viewBox="0 0 24 24" stroke="currentColor" strokeWidth={2}>
- <path strokeLinecap="round" strokeLinejoin="round" d="M5 13l4 4L19 7" />
- </svg>
- <p className="text-green-300">{device.update_message || t('spoolbuddy.settings.updateComplete', 'Update complete!')}</p>
- </div>
- </div>
- )}
- {/* Update error */}
- {device.update_status === 'error' && (
- <div className="rounded-lg p-3 text-sm bg-red-900/30 border border-red-800">
- <p className="text-red-300">{device.update_message || t('spoolbuddy.settings.updateFailed', 'Update failed')}</p>
- </div>
- )}
- {/* Check for updates */}
- <div className="bg-zinc-800 rounded-lg p-4 space-y-3">
- <button
- onClick={checkForUpdates}
- disabled={checking || isUpdating}
- className="w-full px-4 py-2.5 rounded-lg text-sm font-medium bg-zinc-700 text-zinc-200 hover:bg-zinc-600 disabled:opacity-40 transition-colors min-h-[44px] flex items-center justify-center gap-2"
- >
- {checking && (
- <svg className="w-4 h-4 animate-spin" viewBox="0 0 24 24" fill="none">
- <circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4" />
- <path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z" />
- </svg>
- )}
- {checking ? t('spoolbuddy.settings.checking', 'Checking...') : t('spoolbuddy.settings.checkUpdates', 'Check for Updates')}
- </button>
- {/* Error feedback */}
- {error && (
- <div className="rounded-lg p-3 text-sm bg-red-900/30 border border-red-800">
- <p className="text-red-300">{error}</p>
- </div>
- )}
- {/* Result feedback */}
- {updateResult && (
- <div className={`rounded-lg p-3 text-sm ${
- updateResult.update_available
- ? 'bg-green-900/30 border border-green-800'
- : 'bg-zinc-700/50'
- }`}>
- {updateResult.update_available ? (
- <div className="space-y-3">
- <div className="space-y-1">
- <p className="text-green-300 font-medium">
- {t('spoolbuddy.settings.updateAvailable', 'Update available')}: v{updateResult.latest_version}
- </p>
- <p className="text-xs text-zinc-400">
- {displayVersion ? `${displayVersion} → ${updateResult.latest_version}` : ''}
- </p>
- </div>
- <button
- onClick={applyUpdate}
- disabled={applying || isUpdating || !device.online}
- className="w-full px-4 py-2.5 rounded-lg text-sm font-medium bg-green-600 text-white hover:bg-green-700 disabled:opacity-40 transition-colors min-h-[44px] flex items-center justify-center gap-2"
- >
- {applying && (
- <svg className="w-4 h-4 animate-spin" viewBox="0 0 24 24" fill="none">
- <circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4" />
- <path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z" />
- </svg>
- )}
- {!device.online
- ? t('spoolbuddy.settings.deviceOffline', 'Device Offline')
- : t('spoolbuddy.settings.applyUpdate', 'Apply Update')}
- </button>
- </div>
- ) : (
- <div className="flex items-center gap-2">
- <svg className="w-4 h-4 text-green-400 flex-shrink-0" fill="none" viewBox="0 0 24 24" stroke="currentColor" strokeWidth={2}>
- <path strokeLinecap="round" strokeLinejoin="round" d="M5 13l4 4L19 7" />
- </svg>
- <p className="text-zinc-300">{t('spoolbuddy.settings.upToDate', 'Up to date')}</p>
- </div>
- )}
- </div>
- )}
- {/* Include beta toggle */}
- <div className="flex items-center justify-between pt-1">
- <span className="text-xs text-zinc-500">{t('spoolbuddy.settings.includeBeta', 'Include beta versions')}</span>
- <button
- onClick={toggleBeta}
- className={`relative w-10 h-5 rounded-full transition-colors ${
- includeBeta ? 'bg-green-600' : 'bg-zinc-600'
- }`}
- >
- <div className={`absolute top-0.5 w-4 h-4 bg-white rounded-full transition-transform ${
- includeBeta ? 'translate-x-5' : 'translate-x-0.5'
- }`} />
- </button>
- </div>
- </div>
- </div>
- );
- }
- // --- Main Settings Page ---
- type SettingsTab = 'device' | 'display' | 'scale' | 'updates';
- export function SpoolBuddySettingsPage() {
- const { sbState, setDisplayBrightness, setDisplayBlankTimeout } = useOutletContext<SpoolBuddyOutletContext>();
- const { t } = useTranslation();
- const [activeTab, setActiveTab] = useState<SettingsTab>('device');
- const { data: devices = [] } = useQuery({
- queryKey: ['spoolbuddy-devices'],
- queryFn: () => spoolbuddyApi.getDevices(),
- refetchInterval: 10000,
- });
- // Use first device (most common setup) or find one matching current state
- const device = sbState.deviceId
- ? devices.find((d) => d.device_id === sbState.deviceId) ?? devices[0]
- : devices[0];
- const tabs: { id: SettingsTab; label: string }[] = [
- { id: 'device', label: t('spoolbuddy.settings.tabDevice', 'Device') },
- { id: 'display', label: t('spoolbuddy.settings.tabDisplay', 'Display') },
- { id: 'scale', label: t('spoolbuddy.settings.tabScale', 'Scale') },
- { id: 'updates', label: t('spoolbuddy.settings.tabUpdates', 'Updates') },
- ];
- return (
- <div className="h-full flex flex-col p-4">
- <h1 className="text-xl font-semibold text-zinc-100 mb-3">
- {t('spoolbuddy.nav.settings', 'Settings')}
- </h1>
- {/* Tab bar */}
- <div className="flex gap-1 bg-zinc-800/50 rounded-lg p-1 mb-4">
- {tabs.map((tab) => (
- <button
- key={tab.id}
- onClick={() => setActiveTab(tab.id)}
- className={`flex-1 px-2 py-2 rounded-md text-sm font-medium transition-colors min-h-[36px] ${
- activeTab === tab.id
- ? 'bg-zinc-700 text-zinc-100'
- : 'text-zinc-500 hover:text-zinc-300'
- }`}
- >
- {tab.label}
- </button>
- ))}
- </div>
- {/* Content */}
- <div className="flex-1 min-h-0 overflow-y-auto">
- {!device ? (
- <div className="flex items-center justify-center h-32">
- <div className="text-center text-zinc-500">
- <p className="text-sm">{t('spoolbuddy.settings.noDevice', 'No SpoolBuddy device found')}</p>
- </div>
- </div>
- ) : (
- <>
- {activeTab === 'device' && <DeviceTab device={device} />}
- {activeTab === 'display' && (
- <DisplayTab
- device={device}
- onBrightnessChange={setDisplayBrightness}
- onBlankTimeoutChange={setDisplayBlankTimeout}
- />
- )}
- {activeTab === 'scale' && (
- <ScaleTab
- device={device}
- weight={sbState.weight}
- weightStable={sbState.weightStable}
- rawAdc={sbState.rawAdc}
- />
- )}
- {activeTab === 'updates' && <UpdatesTab device={device} />}
- </>
- )}
- </div>
- </div>
- );
- }
|