SpoolBuddySettingsPage.tsx 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786
  1. import { useState, useCallback, useRef, useEffect } from 'react';
  2. import { useQuery } from '@tanstack/react-query';
  3. import { useOutletContext } from 'react-router-dom';
  4. import { useTranslation } from 'react-i18next';
  5. import type { SpoolBuddyOutletContext } from '../../components/spoolbuddy/SpoolBuddyLayout';
  6. import { spoolbuddyApi, type SpoolBuddyDevice, type DaemonUpdateCheck } from '../../api/client';
  7. function formatUptime(seconds: number): string {
  8. if (seconds < 60) return `${seconds}s`;
  9. if (seconds < 3600) return `${Math.floor(seconds / 60)}m`;
  10. const h = Math.floor(seconds / 3600);
  11. const m = Math.floor((seconds % 3600) / 60);
  12. return `${h}h ${m}m`;
  13. }
  14. function formatDateTime(iso: string | null): string {
  15. if (!iso) return '-';
  16. try {
  17. const d = new Date(iso);
  18. return d.toLocaleDateString(undefined, { month: 'short', day: 'numeric' }) + ' ' +
  19. d.toLocaleTimeString(undefined, { hour: '2-digit', minute: '2-digit' });
  20. } catch {
  21. return '-';
  22. }
  23. }
  24. const BLANK_OPTIONS = [
  25. { label: 'Off', value: 0 },
  26. { label: '1m', value: 60 },
  27. { label: '2m', value: 120 },
  28. { label: '5m', value: 300 },
  29. { label: '10m', value: 600 },
  30. { label: '30m', value: 1800 },
  31. ];
  32. // --- Device Tab ---
  33. function DeviceTab({ device }: { device: SpoolBuddyDevice }) {
  34. const { t } = useTranslation();
  35. return (
  36. <div className="space-y-4">
  37. {/* About */}
  38. <div className="bg-zinc-800 rounded-lg p-4">
  39. <div className="flex items-center gap-3 mb-2">
  40. <img src="/img/spoolbuddy_logo_dark_small.png" alt="SpoolBuddy" className="h-7 w-auto" />
  41. </div>
  42. <p className="text-xs text-zinc-500 mb-1">Part of Bambuddy</p>
  43. <span className="text-xs text-zinc-500">github.com/maziggy/bambuddy</span>
  44. </div>
  45. {/* NFC Reader + Device Info side by side */}
  46. <div className="grid grid-cols-2 gap-3">
  47. {/* NFC Reader */}
  48. <div className="bg-zinc-800 rounded-lg p-3">
  49. <h3 className="text-sm font-semibold text-zinc-300 mb-2">
  50. {t('spoolbuddy.settings.nfcReader', 'NFC Reader')}
  51. </h3>
  52. <div className="space-y-1.5 text-xs">
  53. <div className="flex justify-between">
  54. <span className="text-zinc-500">{t('spoolbuddy.settings.type', 'Type')}</span>
  55. <span className="text-zinc-300 font-mono">
  56. {device.nfc_reader_type || 'N/A'}
  57. </span>
  58. </div>
  59. <div className="flex justify-between">
  60. <span className="text-zinc-500">{t('spoolbuddy.settings.connection', 'Connection')}</span>
  61. <span className="text-zinc-300 font-mono">
  62. {device.nfc_connection || 'N/A'}
  63. </span>
  64. </div>
  65. <div className="flex justify-between items-center">
  66. <span className="text-zinc-500">{t('spoolbuddy.status.status', 'Status')}</span>
  67. <div className="flex items-center gap-1.5">
  68. <div className={`w-2 h-2 rounded-full ${
  69. device.nfc_ok ? 'bg-green-500' : device.nfc_reader_type ? 'bg-red-500' : 'bg-zinc-600'
  70. }`} />
  71. <span className={
  72. device.nfc_ok ? 'text-green-400' : device.nfc_reader_type ? 'text-red-400' : 'text-zinc-500'
  73. }>
  74. {device.nfc_ok
  75. ? t('spoolbuddy.status.nfcReady', 'Ready')
  76. : device.nfc_reader_type
  77. ? t('common.error', 'Error')
  78. : t('spoolbuddy.settings.notConnected', 'N/A')}
  79. </span>
  80. </div>
  81. </div>
  82. </div>
  83. </div>
  84. {/* Device Info */}
  85. <div className="bg-zinc-800 rounded-lg p-3">
  86. <h3 className="text-sm font-semibold text-zinc-300 mb-2">
  87. {t('spoolbuddy.settings.deviceInfo', 'Device Info')}
  88. </h3>
  89. <div className="space-y-1.5 text-xs">
  90. <div className="flex justify-between">
  91. <span className="text-zinc-500">{t('spoolbuddy.settings.hostname', 'Host')}</span>
  92. <span className="text-zinc-300 truncate ml-2">{device.hostname}</span>
  93. </div>
  94. <div className="flex justify-between">
  95. <span className="text-zinc-500">IP</span>
  96. <span className="text-zinc-300">{device.ip_address}</span>
  97. </div>
  98. <div className="flex justify-between">
  99. <span className="text-zinc-500">{t('spoolbuddy.settings.uptime', 'Uptime')}</span>
  100. <span className="text-zinc-300">{formatUptime(device.uptime_s)}</span>
  101. </div>
  102. <div className="flex justify-between items-center">
  103. <span className="text-zinc-500">{t('spoolbuddy.status.status', 'Status')}</span>
  104. <div className="flex items-center gap-1.5">
  105. <div className={`w-2 h-2 rounded-full ${device.online ? 'bg-green-500' : 'bg-zinc-600'}`} />
  106. <span className={device.online ? 'text-green-400' : 'text-zinc-500'}>
  107. {device.online ? t('spoolbuddy.status.online', 'Online') : t('spoolbuddy.status.offline', 'Offline')}
  108. </span>
  109. </div>
  110. </div>
  111. </div>
  112. </div>
  113. </div>
  114. {/* Device ID (full width, below cards) */}
  115. <div className="bg-zinc-800 rounded-lg px-3 py-2 flex justify-between items-center text-xs">
  116. <span className="text-zinc-500">Device ID</span>
  117. <span className="text-zinc-400 font-mono">{device.device_id}</span>
  118. </div>
  119. </div>
  120. );
  121. }
  122. // --- Display Tab ---
  123. function DisplayTab({ device, onBrightnessChange, onBlankTimeoutChange }: {
  124. device: SpoolBuddyDevice;
  125. onBrightnessChange: (value: number) => void;
  126. onBlankTimeoutChange: (value: number) => void;
  127. }) {
  128. const { t } = useTranslation();
  129. const [brightness, setBrightness] = useState(device.display_brightness);
  130. const [blankTimeout, setBlankTimeout] = useState(device.display_blank_timeout);
  131. const [saved, setSaved] = useState(false);
  132. const debounceRef = useRef<ReturnType<typeof setTimeout>>(undefined);
  133. const savedTimerRef = useRef<ReturnType<typeof setTimeout>>(undefined);
  134. // Sync local state when device data updates from server
  135. useEffect(() => {
  136. setBrightness(device.display_brightness);
  137. setBlankTimeout(device.display_blank_timeout);
  138. }, [device.display_brightness, device.display_blank_timeout]);
  139. const showSaved = useCallback(() => {
  140. setSaved(true);
  141. if (savedTimerRef.current) clearTimeout(savedTimerRef.current);
  142. savedTimerRef.current = setTimeout(() => setSaved(false), 1500);
  143. }, []);
  144. const sendDisplayUpdate = useCallback((b: number, bt: number) => {
  145. if (debounceRef.current) clearTimeout(debounceRef.current);
  146. debounceRef.current = setTimeout(() => {
  147. spoolbuddyApi.updateDisplay(device.device_id, b, bt)
  148. .then(() => showSaved())
  149. .catch((e) => console.error('Failed to update display:', e));
  150. }, 300);
  151. }, [device.device_id, showSaved]);
  152. const handleBrightnessChange = (value: number) => {
  153. setBrightness(value);
  154. onBrightnessChange(value); // Instant layout update
  155. sendDisplayUpdate(value, blankTimeout);
  156. };
  157. const handleBlankTimeoutChange = (value: number) => {
  158. setBlankTimeout(value);
  159. onBlankTimeoutChange(value); // Instant layout update
  160. sendDisplayUpdate(brightness, value);
  161. };
  162. return (
  163. <div className="space-y-4">
  164. {/* Brightness */}
  165. <div className="bg-zinc-800 rounded-lg p-4">
  166. <div className="flex items-center justify-between mb-3">
  167. <h3 className="text-sm font-semibold text-zinc-300">
  168. {t('spoolbuddy.settings.brightness', 'Brightness')}
  169. </h3>
  170. {saved && (
  171. <span className="text-xs text-green-400 flex items-center gap-1 animate-pulse">
  172. <svg className="w-3 h-3" fill="none" viewBox="0 0 24 24" stroke="currentColor" strokeWidth={3}>
  173. <path strokeLinecap="round" strokeLinejoin="round" d="M5 13l4 4L19 7" />
  174. </svg>
  175. {t('spoolbuddy.settings.saved', 'Saved')}
  176. </span>
  177. )}
  178. </div>
  179. <div className="flex items-center gap-3">
  180. <svg className="w-4 h-4 text-zinc-500 flex-shrink-0" fill="none" viewBox="0 0 24 24" stroke="currentColor" strokeWidth={2}>
  181. <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" />
  182. </svg>
  183. <input
  184. type="range"
  185. min={0}
  186. max={100}
  187. value={brightness}
  188. onChange={(e) => handleBrightnessChange(parseInt(e.target.value))}
  189. className="flex-1 h-2 bg-zinc-700 rounded-lg appearance-none cursor-pointer accent-green-500"
  190. />
  191. <span className="text-sm font-mono text-zinc-400 w-10 text-right">{brightness}%</span>
  192. </div>
  193. {!device.has_backlight && (
  194. <p className="text-xs text-zinc-600 mt-2">
  195. {t('spoolbuddy.settings.noBacklight', 'No DSI backlight detected. Brightness control requires a DSI display.')}
  196. </p>
  197. )}
  198. </div>
  199. {/* Screen blank timeout */}
  200. <div className="bg-zinc-800 rounded-lg p-4">
  201. <h3 className="text-sm font-semibold text-zinc-300 mb-1">
  202. {t('spoolbuddy.settings.screenBlank', 'Screen Blank Timeout')}
  203. </h3>
  204. <p className="text-xs text-zinc-500 mb-3">
  205. {t('spoolbuddy.settings.screenBlankDesc', 'Screen turns off after inactivity. Touch to wake.')}
  206. </p>
  207. <div className="grid grid-cols-3 gap-2">
  208. {BLANK_OPTIONS.map((opt) => (
  209. <button
  210. key={opt.value}
  211. onClick={() => handleBlankTimeoutChange(opt.value)}
  212. className={`px-3 py-2 rounded-lg text-sm font-medium transition-colors min-h-[40px] ${
  213. blankTimeout === opt.value
  214. ? 'bg-green-600 text-white'
  215. : 'bg-zinc-700 text-zinc-300 hover:bg-zinc-600'
  216. }`}
  217. >
  218. {opt.label}
  219. </button>
  220. ))}
  221. </div>
  222. </div>
  223. <p className="text-xs text-zinc-600 text-center">
  224. {t('spoolbuddy.settings.displayNote', 'Brightness is applied as a software filter.')}
  225. </p>
  226. </div>
  227. );
  228. }
  229. // --- Scale Tab ---
  230. function StepIndicator({ step, labels }: { step: 'tare' | 'weight'; labels: { tare: string; weight: string } }) {
  231. return (
  232. <div className="flex flex-col items-center w-16 shrink-0 pt-1">
  233. {/* Step 1 circle */}
  234. <div className={`flex items-center justify-center w-7 h-7 rounded-full text-xs font-bold ${
  235. step === 'tare'
  236. ? 'bg-green-600 text-white'
  237. : 'bg-green-600/20 text-green-400'
  238. }`}>
  239. {step === 'weight' ? (
  240. <svg className="w-3.5 h-3.5" fill="none" viewBox="0 0 24 24" stroke="currentColor" strokeWidth={3}>
  241. <path strokeLinecap="round" strokeLinejoin="round" d="M5 13l4 4L19 7" />
  242. </svg>
  243. ) : '1'}
  244. </div>
  245. <span className={`text-[10px] mt-0.5 ${step === 'tare' ? 'text-green-400 font-medium' : 'text-green-400/60'}`}>
  246. {labels.tare}
  247. </span>
  248. {/* Connector line */}
  249. <div className={`w-px h-5 my-1 ${step === 'weight' ? 'bg-green-600/40' : 'bg-zinc-700'}`} />
  250. {/* Step 2 circle */}
  251. <div className={`flex items-center justify-center w-7 h-7 rounded-full text-xs font-bold ${
  252. step === 'weight'
  253. ? 'bg-green-600 text-white'
  254. : 'bg-zinc-700 text-zinc-500'
  255. }`}>
  256. 2
  257. </div>
  258. <span className={`text-[10px] mt-0.5 ${step === 'weight' ? 'text-green-400 font-medium' : 'text-zinc-600'}`}>
  259. {labels.weight}
  260. </span>
  261. </div>
  262. );
  263. }
  264. function ScaleTab({ device, weight, weightStable, rawAdc }: {
  265. device: SpoolBuddyDevice;
  266. weight: number | null;
  267. weightStable: boolean;
  268. rawAdc: number | null;
  269. }) {
  270. const { t } = useTranslation();
  271. const [calStep, setCalStep] = useState<'idle' | 'tare' | 'weight'>('idle');
  272. const [knownWeight, setKnownWeight] = useState('500');
  273. const [tareRawAdc, setTareRawAdc] = useState<number | null>(null);
  274. const [busy, setBusy] = useState(false);
  275. const [status, setStatus] = useState<{ type: 'ok' | 'error'; msg: string } | null>(null);
  276. const numpadPress = (key: string) => {
  277. if (key === 'backspace') {
  278. setKnownWeight((v) => v.slice(0, -1) || '');
  279. } else if (key === '.' && !knownWeight.includes('.')) {
  280. setKnownWeight((v) => v + '.');
  281. } else if (key >= '0' && key <= '9') {
  282. setKnownWeight((v) => (v === '0' ? key : v + key));
  283. }
  284. };
  285. const handleTare = async () => {
  286. setBusy(true);
  287. setStatus(null);
  288. try {
  289. await spoolbuddyApi.tare(device.device_id);
  290. setStatus({ type: 'ok', msg: t('spoolbuddy.settings.tareSet', 'Tare command sent. Waiting for device...') });
  291. } catch {
  292. setStatus({ type: 'error', msg: t('spoolbuddy.settings.tareFailed', 'Failed to send tare command') });
  293. } finally {
  294. setBusy(false);
  295. }
  296. };
  297. const handleCalStep = async () => {
  298. if (calStep === 'tare') {
  299. setBusy(true);
  300. setStatus(null);
  301. try {
  302. setTareRawAdc(rawAdc);
  303. await spoolbuddyApi.tare(device.device_id);
  304. setStatus({ type: 'ok', msg: t('spoolbuddy.settings.zeroSet', 'Zero point set. Place known weight on scale.') });
  305. setCalStep('weight');
  306. } catch {
  307. setStatus({ type: 'error', msg: t('spoolbuddy.settings.tareFailed', 'Failed to send tare command') });
  308. } finally {
  309. setBusy(false);
  310. }
  311. } else if (calStep === 'weight') {
  312. const weightNum = parseFloat(knownWeight);
  313. if (rawAdc === null || !weightNum || weightNum <= 0) return;
  314. setBusy(true);
  315. setStatus(null);
  316. try {
  317. await spoolbuddyApi.setCalibrationFactor(device.device_id, weightNum, rawAdc, tareRawAdc ?? undefined);
  318. setStatus({ type: 'ok', msg: t('spoolbuddy.settings.calibrationDone', 'Calibration complete!') });
  319. setCalStep('idle');
  320. } catch {
  321. setStatus({ type: 'error', msg: t('spoolbuddy.settings.calibrationFailed', 'Calibration failed') });
  322. } finally {
  323. setBusy(false);
  324. }
  325. }
  326. };
  327. // --- Idle state: weight card + buttons ---
  328. if (calStep === 'idle') {
  329. return (
  330. <div className="flex flex-col h-full">
  331. {/* Weight + info card */}
  332. <div className="bg-zinc-800 rounded-lg p-3 mb-3">
  333. <div className="flex items-center justify-between">
  334. <div className="flex items-center gap-2">
  335. <div className={`w-2 h-2 rounded-full ${weightStable ? 'bg-green-500' : 'bg-amber-500 animate-pulse'}`} />
  336. <span className="text-lg font-mono text-zinc-200">
  337. {weight !== null ? `${weight.toFixed(1)} g` : '-- g'}
  338. </span>
  339. </div>
  340. <div className="text-xs text-zinc-500 text-right">
  341. <span>{t('spoolbuddy.settings.tareOffset', 'Tare')}: {device.tare_offset}</span>
  342. <span className="mx-1.5">&middot;</span>
  343. <span>{t('spoolbuddy.settings.calFactor', 'Factor')}: {device.calibration_factor.toFixed(2)}</span>
  344. </div>
  345. </div>
  346. {device.last_calibrated_at && (
  347. <div className="text-xs text-zinc-600 mt-1">
  348. {t('spoolbuddy.settings.lastCalibrated', 'Last calibrated')}: {formatDateTime(device.last_calibrated_at)}
  349. </div>
  350. )}
  351. </div>
  352. {/* Status message */}
  353. {status && (
  354. <div className={`rounded-lg px-3 py-2 mb-3 text-sm ${
  355. 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'
  356. }`}>
  357. {status.msg}
  358. </div>
  359. )}
  360. {/* Action buttons */}
  361. <div className="flex gap-2">
  362. <button
  363. onClick={handleTare}
  364. disabled={busy}
  365. 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"
  366. >
  367. {busy && (
  368. <svg className="w-4 h-4 animate-spin" viewBox="0 0 24 24" fill="none">
  369. <circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4" />
  370. <path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z" />
  371. </svg>
  372. )}
  373. {t('spoolbuddy.weight.tare', 'Tare')}
  374. </button>
  375. <button
  376. onClick={() => { setCalStep('tare'); setStatus(null); }}
  377. 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]"
  378. >
  379. {t('spoolbuddy.weight.calibrate', 'Calibrate')}
  380. </button>
  381. </div>
  382. </div>
  383. );
  384. }
  385. // --- Calibration wizard: step indicator left + content right ---
  386. return (
  387. <div className="flex gap-3">
  388. {/* Left: step indicator */}
  389. <StepIndicator step={calStep} labels={{ tare: t('spoolbuddy.weight.tare', 'Tare'), weight: t('spoolbuddy.settings.knownWeight', 'Known weight') }} />
  390. {/* Right: content */}
  391. <div className="flex-1 min-w-0">
  392. {/* Live weight bar */}
  393. <div className="flex items-center gap-2 bg-zinc-800 rounded-lg px-3 py-1.5 mb-1.5">
  394. <div className={`w-2 h-2 rounded-full shrink-0 ${weightStable ? 'bg-green-500' : 'bg-amber-500 animate-pulse'}`} />
  395. <span className="text-sm font-mono text-zinc-200">
  396. {weight !== null ? `${weight.toFixed(1)} g` : '-- g'}
  397. </span>
  398. <span className={`text-xs ml-auto ${weightStable ? 'text-green-400' : 'text-amber-400'}`}>
  399. {weightStable ? t('spoolbuddy.settings.stable', 'Stable') : t('spoolbuddy.settings.settling', 'Settling...')}
  400. </span>
  401. </div>
  402. {/* Status message */}
  403. {status && (
  404. <div className={`rounded-lg px-3 py-1.5 mb-1.5 text-xs ${
  405. 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'
  406. }`}>
  407. {status.msg}
  408. </div>
  409. )}
  410. {/* Step content */}
  411. {calStep === 'tare' ? (
  412. <p className="text-sm text-zinc-300 mb-3">
  413. {t('spoolbuddy.settings.calStep1', 'Remove all items from the scale and press Set Zero.')}
  414. </p>
  415. ) : (
  416. <>
  417. <div className="flex items-center gap-2 mb-1.5">
  418. <span className="text-xs text-zinc-400 shrink-0">{t('spoolbuddy.settings.knownWeight', 'Known weight')}</span>
  419. <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">
  420. {knownWeight || '0'}<span className="text-zinc-500 ml-1">g</span>
  421. </div>
  422. </div>
  423. <div className="grid grid-cols-4 gap-1 mb-1.5">
  424. {['7','8','9','backspace','4','5','6','.','1','2','3','0'].map((key) => (
  425. <button
  426. key={key}
  427. onClick={() => numpadPress(key)}
  428. className={`rounded text-lg font-medium transition-colors h-[48px] active:scale-95 ${
  429. key === 'backspace'
  430. ? 'bg-zinc-700 text-zinc-300 hover:bg-zinc-600'
  431. : 'bg-zinc-800 text-zinc-100 hover:bg-zinc-700 border border-zinc-700'
  432. }`}
  433. >
  434. {key === 'backspace' ? '\u232B' : key}
  435. </button>
  436. ))}
  437. </div>
  438. </>
  439. )}
  440. {/* Action buttons */}
  441. <div className="flex gap-2">
  442. <button
  443. onClick={() => { setCalStep('idle'); setStatus(null); }}
  444. 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]"
  445. >
  446. {t('common.cancel', 'Cancel')}
  447. </button>
  448. <button
  449. onClick={handleCalStep}
  450. disabled={busy}
  451. 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"
  452. >
  453. {busy && (
  454. <svg className="w-4 h-4 animate-spin" viewBox="0 0 24 24" fill="none">
  455. <circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4" />
  456. <path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z" />
  457. </svg>
  458. )}
  459. {calStep === 'tare' ? t('spoolbuddy.settings.setZero', 'Set Zero') : t('spoolbuddy.settings.calibrateNow', 'Calibrate')}
  460. </button>
  461. </div>
  462. </div>
  463. </div>
  464. );
  465. }
  466. // --- Updates Tab ---
  467. function UpdatesTab({ device }: { device: SpoolBuddyDevice }) {
  468. const { t } = useTranslation();
  469. const [checking, setChecking] = useState(false);
  470. const [applying, setApplying] = useState(false);
  471. const [updateResult, setUpdateResult] = useState<DaemonUpdateCheck | null>(null);
  472. const [error, setError] = useState<string | null>(null);
  473. const [includeBeta, setIncludeBeta] = useState(() => {
  474. try {
  475. return localStorage.getItem('spoolbuddy-include-beta') === 'true';
  476. } catch {
  477. return false;
  478. }
  479. });
  480. const isUpdating = device.update_status === 'pending' || device.update_status === 'updating';
  481. const toggleBeta = () => {
  482. const next = !includeBeta;
  483. setIncludeBeta(next);
  484. try {
  485. localStorage.setItem('spoolbuddy-include-beta', String(next));
  486. } catch {
  487. // localStorage unavailable
  488. }
  489. setUpdateResult(null);
  490. setError(null);
  491. };
  492. const checkForUpdates = async () => {
  493. setChecking(true);
  494. setUpdateResult(null);
  495. setError(null);
  496. try {
  497. const result = await spoolbuddyApi.checkDaemonUpdate(device.device_id, includeBeta);
  498. setUpdateResult(result);
  499. } catch (e) {
  500. setError(e instanceof Error ? e.message : 'Failed to check for updates');
  501. } finally {
  502. setChecking(false);
  503. }
  504. };
  505. const applyUpdate = async () => {
  506. setApplying(true);
  507. setError(null);
  508. try {
  509. await spoolbuddyApi.triggerUpdate(device.device_id);
  510. } catch (e) {
  511. setError(e instanceof Error ? e.message : 'Failed to trigger update');
  512. } finally {
  513. setApplying(false);
  514. }
  515. };
  516. // Show version from device, or from update check result if available
  517. const displayVersion = device.firmware_version
  518. || (updateResult?.current_version && updateResult.current_version !== '0.0.0' ? updateResult.current_version : null);
  519. return (
  520. <div className="space-y-4">
  521. {/* Current version */}
  522. <div className="bg-zinc-800 rounded-lg p-4">
  523. <h3 className="text-sm font-semibold text-zinc-300 mb-3">
  524. {t('spoolbuddy.settings.daemonVersion', 'Daemon Version')}
  525. </h3>
  526. <div className="flex justify-between items-center text-sm">
  527. <span className="text-zinc-500">{t('spoolbuddy.settings.currentVersion', 'Current')}</span>
  528. <span className="text-zinc-200 font-mono">
  529. {displayVersion || (
  530. <span className="text-zinc-500 italic">{t('spoolbuddy.settings.versionPending', 'Waiting for daemon...')}</span>
  531. )}
  532. </span>
  533. </div>
  534. </div>
  535. {/* Update progress (shown when update is in progress) */}
  536. {isUpdating && (
  537. <div className="bg-zinc-800 rounded-lg p-4">
  538. <div className="flex items-center gap-3">
  539. <svg className="w-5 h-5 animate-spin text-green-400 flex-shrink-0" viewBox="0 0 24 24" fill="none">
  540. <circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4" />
  541. <path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z" />
  542. </svg>
  543. <div>
  544. <p className="text-sm font-medium text-green-300">
  545. {t('spoolbuddy.settings.updating', 'Updating...')}
  546. </p>
  547. <p className="text-xs text-zinc-400 mt-0.5">
  548. {device.update_message || t('spoolbuddy.settings.updateWaiting', 'Waiting for device...')}
  549. </p>
  550. </div>
  551. </div>
  552. </div>
  553. )}
  554. {/* Update complete */}
  555. {device.update_status === 'complete' && (
  556. <div className="rounded-lg p-3 text-sm bg-green-900/30 border border-green-800">
  557. <div className="flex items-center gap-2">
  558. <svg className="w-4 h-4 text-green-400 flex-shrink-0" fill="none" viewBox="0 0 24 24" stroke="currentColor" strokeWidth={2}>
  559. <path strokeLinecap="round" strokeLinejoin="round" d="M5 13l4 4L19 7" />
  560. </svg>
  561. <p className="text-green-300">{device.update_message || t('spoolbuddy.settings.updateComplete', 'Update complete!')}</p>
  562. </div>
  563. </div>
  564. )}
  565. {/* Update error */}
  566. {device.update_status === 'error' && (
  567. <div className="rounded-lg p-3 text-sm bg-red-900/30 border border-red-800">
  568. <p className="text-red-300">{device.update_message || t('spoolbuddy.settings.updateFailed', 'Update failed')}</p>
  569. </div>
  570. )}
  571. {/* Check for updates */}
  572. <div className="bg-zinc-800 rounded-lg p-4 space-y-3">
  573. <button
  574. onClick={checkForUpdates}
  575. disabled={checking || isUpdating}
  576. 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"
  577. >
  578. {checking && (
  579. <svg className="w-4 h-4 animate-spin" viewBox="0 0 24 24" fill="none">
  580. <circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4" />
  581. <path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z" />
  582. </svg>
  583. )}
  584. {checking ? t('spoolbuddy.settings.checking', 'Checking...') : t('spoolbuddy.settings.checkUpdates', 'Check for Updates')}
  585. </button>
  586. {/* Error feedback */}
  587. {error && (
  588. <div className="rounded-lg p-3 text-sm bg-red-900/30 border border-red-800">
  589. <p className="text-red-300">{error}</p>
  590. </div>
  591. )}
  592. {/* Result feedback */}
  593. {updateResult && (
  594. <div className={`rounded-lg p-3 text-sm ${
  595. updateResult.update_available
  596. ? 'bg-green-900/30 border border-green-800'
  597. : 'bg-zinc-700/50'
  598. }`}>
  599. {updateResult.update_available ? (
  600. <div className="space-y-3">
  601. <div className="space-y-1">
  602. <p className="text-green-300 font-medium">
  603. {t('spoolbuddy.settings.updateAvailable', 'Update available')}: v{updateResult.latest_version}
  604. </p>
  605. <p className="text-xs text-zinc-400">
  606. {displayVersion ? `${displayVersion} → ${updateResult.latest_version}` : ''}
  607. </p>
  608. </div>
  609. <button
  610. onClick={applyUpdate}
  611. disabled={applying || isUpdating || !device.online}
  612. 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"
  613. >
  614. {applying && (
  615. <svg className="w-4 h-4 animate-spin" viewBox="0 0 24 24" fill="none">
  616. <circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4" />
  617. <path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z" />
  618. </svg>
  619. )}
  620. {!device.online
  621. ? t('spoolbuddy.settings.deviceOffline', 'Device Offline')
  622. : t('spoolbuddy.settings.applyUpdate', 'Apply Update')}
  623. </button>
  624. </div>
  625. ) : (
  626. <div className="flex items-center gap-2">
  627. <svg className="w-4 h-4 text-green-400 flex-shrink-0" fill="none" viewBox="0 0 24 24" stroke="currentColor" strokeWidth={2}>
  628. <path strokeLinecap="round" strokeLinejoin="round" d="M5 13l4 4L19 7" />
  629. </svg>
  630. <p className="text-zinc-300">{t('spoolbuddy.settings.upToDate', 'Up to date')}</p>
  631. </div>
  632. )}
  633. </div>
  634. )}
  635. {/* Include beta toggle */}
  636. <div className="flex items-center justify-between pt-1">
  637. <span className="text-xs text-zinc-500">{t('spoolbuddy.settings.includeBeta', 'Include beta versions')}</span>
  638. <button
  639. onClick={toggleBeta}
  640. className={`relative w-10 h-5 rounded-full transition-colors ${
  641. includeBeta ? 'bg-green-600' : 'bg-zinc-600'
  642. }`}
  643. >
  644. <div className={`absolute top-0.5 w-4 h-4 bg-white rounded-full transition-transform ${
  645. includeBeta ? 'translate-x-5' : 'translate-x-0.5'
  646. }`} />
  647. </button>
  648. </div>
  649. </div>
  650. </div>
  651. );
  652. }
  653. // --- Main Settings Page ---
  654. type SettingsTab = 'device' | 'display' | 'scale' | 'updates';
  655. export function SpoolBuddySettingsPage() {
  656. const { sbState, setDisplayBrightness, setDisplayBlankTimeout } = useOutletContext<SpoolBuddyOutletContext>();
  657. const { t } = useTranslation();
  658. const [activeTab, setActiveTab] = useState<SettingsTab>('device');
  659. const { data: devices = [] } = useQuery({
  660. queryKey: ['spoolbuddy-devices'],
  661. queryFn: () => spoolbuddyApi.getDevices(),
  662. refetchInterval: 10000,
  663. });
  664. // Use first device (most common setup) or find one matching current state
  665. const device = sbState.deviceId
  666. ? devices.find((d) => d.device_id === sbState.deviceId) ?? devices[0]
  667. : devices[0];
  668. const tabs: { id: SettingsTab; label: string }[] = [
  669. { id: 'device', label: t('spoolbuddy.settings.tabDevice', 'Device') },
  670. { id: 'display', label: t('spoolbuddy.settings.tabDisplay', 'Display') },
  671. { id: 'scale', label: t('spoolbuddy.settings.tabScale', 'Scale') },
  672. { id: 'updates', label: t('spoolbuddy.settings.tabUpdates', 'Updates') },
  673. ];
  674. return (
  675. <div className="h-full flex flex-col p-4">
  676. <h1 className="text-xl font-semibold text-zinc-100 mb-3">
  677. {t('spoolbuddy.nav.settings', 'Settings')}
  678. </h1>
  679. {/* Tab bar */}
  680. <div className="flex gap-1 bg-zinc-800/50 rounded-lg p-1 mb-4">
  681. {tabs.map((tab) => (
  682. <button
  683. key={tab.id}
  684. onClick={() => setActiveTab(tab.id)}
  685. className={`flex-1 px-2 py-2 rounded-md text-sm font-medium transition-colors min-h-[36px] ${
  686. activeTab === tab.id
  687. ? 'bg-zinc-700 text-zinc-100'
  688. : 'text-zinc-500 hover:text-zinc-300'
  689. }`}
  690. >
  691. {tab.label}
  692. </button>
  693. ))}
  694. </div>
  695. {/* Content */}
  696. <div className="flex-1 min-h-0 overflow-y-auto">
  697. {!device ? (
  698. <div className="flex items-center justify-center h-32">
  699. <div className="text-center text-zinc-500">
  700. <p className="text-sm">{t('spoolbuddy.settings.noDevice', 'No SpoolBuddy device found')}</p>
  701. </div>
  702. </div>
  703. ) : (
  704. <>
  705. {activeTab === 'device' && <DeviceTab device={device} />}
  706. {activeTab === 'display' && (
  707. <DisplayTab
  708. device={device}
  709. onBrightnessChange={setDisplayBrightness}
  710. onBlankTimeoutChange={setDisplayBlankTimeout}
  711. />
  712. )}
  713. {activeTab === 'scale' && (
  714. <ScaleTab
  715. device={device}
  716. weight={sbState.weight}
  717. weightStable={sbState.weightStable}
  718. rawAdc={sbState.rawAdc}
  719. />
  720. )}
  721. {activeTab === 'updates' && <UpdatesTab device={device} />}
  722. </>
  723. )}
  724. </div>
  725. </div>
  726. );
  727. }