VirtualPrinterSettings.tsx 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544
  1. import { useState, useEffect } from 'react';
  2. import { useTranslation } from 'react-i18next';
  3. import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
  4. import { Loader2, Check, AlertTriangle, Printer, Eye, EyeOff, Info, ChevronDown, ExternalLink, ArrowRightLeft } from 'lucide-react';
  5. import { api, virtualPrinterApi } from '../api/client';
  6. import { Card, CardContent, CardHeader } from './Card';
  7. import { Button } from './Button';
  8. import { useToast } from '../contexts/ToastContext';
  9. type LocalMode = 'immediate' | 'review' | 'print_queue' | 'proxy';
  10. export function VirtualPrinterSettings() {
  11. const { t } = useTranslation();
  12. const queryClient = useQueryClient();
  13. const { showToast } = useToast();
  14. const [localEnabled, setLocalEnabled] = useState(false);
  15. const [localAccessCode, setLocalAccessCode] = useState('');
  16. const [localMode, setLocalMode] = useState<LocalMode>('immediate');
  17. const [localModel, setLocalModel] = useState('3DPrinter-X1-Carbon');
  18. const [localTargetPrinterId, setLocalTargetPrinterId] = useState<number | null>(null);
  19. const [showAccessCode, setShowAccessCode] = useState(false);
  20. const [pendingAction, setPendingAction] = useState<'toggle' | 'accessCode' | 'mode' | 'model' | 'targetPrinter' | null>(null);
  21. // Fetch current settings
  22. const { data: settings, isLoading } = useQuery({
  23. queryKey: ['virtual-printer-settings'],
  24. queryFn: virtualPrinterApi.getSettings,
  25. refetchInterval: 10000, // Refresh every 10 seconds for status updates
  26. });
  27. // Fetch available models
  28. const { data: modelsData } = useQuery({
  29. queryKey: ['virtual-printer-models'],
  30. queryFn: virtualPrinterApi.getModels,
  31. });
  32. // Fetch printers for proxy mode dropdown
  33. const { data: printers } = useQuery({
  34. queryKey: ['printers'],
  35. queryFn: api.getPrinters,
  36. });
  37. // Initialize local state from settings
  38. useEffect(() => {
  39. if (settings) {
  40. setLocalEnabled(settings.enabled);
  41. // Map legacy 'queue' mode to 'review'
  42. let mode: LocalMode = settings.mode === 'queue' ? 'review' : settings.mode as LocalMode;
  43. if (mode !== 'immediate' && mode !== 'review' && mode !== 'print_queue' && mode !== 'proxy') {
  44. mode = 'immediate'; // fallback
  45. }
  46. setLocalMode(mode);
  47. setLocalModel(settings.model);
  48. setLocalTargetPrinterId(settings.target_printer_id);
  49. }
  50. }, [settings]);
  51. // Update mutation
  52. const updateMutation = useMutation({
  53. mutationFn: (data: { enabled?: boolean; access_code?: string; mode?: LocalMode; model?: string; target_printer_id?: number }) =>
  54. virtualPrinterApi.updateSettings(data),
  55. onSuccess: () => {
  56. queryClient.invalidateQueries({ queryKey: ['virtual-printer-settings'] });
  57. showToast(t('virtualPrinter.toast.updated'));
  58. setPendingAction(null);
  59. },
  60. onError: (error: Error) => {
  61. showToast(error.message || t('virtualPrinter.toast.failedToUpdate'), 'error');
  62. // Revert local state on error
  63. if (settings) {
  64. setLocalEnabled(settings.enabled);
  65. // Map legacy 'queue' mode to 'review'
  66. const mode = settings.mode === 'queue' ? 'review' : settings.mode;
  67. setLocalMode(['immediate', 'review', 'print_queue', 'proxy'].includes(mode) ? mode as LocalMode : 'immediate');
  68. setLocalModel(settings.model);
  69. setLocalTargetPrinterId(settings.target_printer_id);
  70. }
  71. setPendingAction(null);
  72. },
  73. });
  74. const handleToggleEnabled = () => {
  75. const newEnabled = !localEnabled;
  76. // Validation depends on mode
  77. if (newEnabled) {
  78. if (localMode === 'proxy') {
  79. // Proxy mode requires target printer
  80. if (!localTargetPrinterId) {
  81. showToast(t('virtualPrinter.toast.targetPrinterRequired'), 'error');
  82. return;
  83. }
  84. } else {
  85. // Other modes require access code
  86. if (!localAccessCode && !settings?.access_code_set) {
  87. showToast(t('virtualPrinter.toast.accessCodeRequired'), 'error');
  88. return;
  89. }
  90. }
  91. }
  92. setLocalEnabled(newEnabled);
  93. setPendingAction('toggle');
  94. updateMutation.mutate({
  95. enabled: newEnabled,
  96. access_code: localMode !== 'proxy' ? (localAccessCode || undefined) : undefined,
  97. mode: localMode,
  98. target_printer_id: localMode === 'proxy' ? (localTargetPrinterId ?? undefined) : undefined,
  99. });
  100. };
  101. const handleAccessCodeChange = () => {
  102. if (!localAccessCode) {
  103. showToast(t('virtualPrinter.toast.accessCodeEmpty'), 'error');
  104. return;
  105. }
  106. if (localAccessCode.length !== 8) {
  107. showToast(t('virtualPrinter.toast.accessCodeLength'), 'error');
  108. return;
  109. }
  110. setPendingAction('accessCode');
  111. updateMutation.mutate({
  112. access_code: localAccessCode,
  113. });
  114. setLocalAccessCode(''); // Clear after saving
  115. };
  116. const handleModeChange = (mode: LocalMode) => {
  117. setLocalMode(mode);
  118. setPendingAction('mode');
  119. updateMutation.mutate({ mode });
  120. };
  121. const handleTargetPrinterChange = (printerId: number) => {
  122. setLocalTargetPrinterId(printerId);
  123. setPendingAction('targetPrinter');
  124. updateMutation.mutate({
  125. target_printer_id: printerId,
  126. });
  127. };
  128. const handleModelChange = (model: string) => {
  129. setLocalModel(model);
  130. setPendingAction('model');
  131. updateMutation.mutate({ model });
  132. };
  133. if (isLoading) {
  134. return (
  135. <Card>
  136. <CardContent className="py-8 flex justify-center">
  137. <Loader2 className="w-6 h-6 animate-spin text-bambu-green" />
  138. </CardContent>
  139. </Card>
  140. );
  141. }
  142. const status = settings?.status;
  143. const isRunning = status?.running || false;
  144. return (
  145. <div className="flex flex-col lg:flex-row gap-6 lg:gap-8">
  146. {/* Left Column - Settings */}
  147. <div className="space-y-6 lg:w-[480px] lg:flex-shrink-0">
  148. <Card>
  149. <CardHeader>
  150. <div className="flex items-center justify-between">
  151. <div className="flex items-center gap-2">
  152. <Printer className="w-5 h-5 text-bambu-green" />
  153. <h2 className="text-lg font-semibold text-white">{t('virtualPrinter.title')}</h2>
  154. </div>
  155. {status && (
  156. <div className={`flex items-center gap-2 text-sm ${isRunning ? 'text-green-400' : 'text-bambu-gray'}`}>
  157. <span className={`w-2 h-2 rounded-full ${isRunning ? 'bg-green-400 animate-pulse' : 'bg-gray-500'}`} />
  158. {isRunning ? t('virtualPrinter.running') : t('virtualPrinter.stopped')}
  159. </div>
  160. )}
  161. </div>
  162. </CardHeader>
  163. <CardContent className="space-y-4">
  164. <p className="text-sm text-bambu-gray">
  165. {localMode === 'proxy'
  166. ? t('virtualPrinter.description.proxy')
  167. : t('virtualPrinter.description.default')}
  168. </p>
  169. {/* Enable/Disable Toggle */}
  170. <div className="flex items-center justify-between py-3 border-t border-bambu-dark-tertiary">
  171. <div>
  172. <div className="text-white font-medium">{t('virtualPrinter.enable.title')}</div>
  173. <div className="text-sm text-bambu-gray">
  174. {isRunning ? (
  175. localMode === 'proxy'
  176. ? t('virtualPrinter.enable.proxyingTo', { name: printers?.find(p => p.id === localTargetPrinterId)?.name || 'printer' })
  177. : t('virtualPrinter.enable.visibleInSlicer')
  178. ) : t('virtualPrinter.enable.notActive')}
  179. </div>
  180. </div>
  181. <button
  182. onClick={handleToggleEnabled}
  183. disabled={pendingAction === 'toggle'}
  184. className={`relative w-12 h-6 rounded-full transition-colors ${
  185. localEnabled ? 'bg-bambu-green' : 'bg-bambu-dark-tertiary'
  186. } ${pendingAction === 'toggle' ? 'opacity-50' : ''}`}
  187. >
  188. <span
  189. className={`absolute top-1 left-1 w-4 h-4 bg-white rounded-full transition-transform ${
  190. localEnabled ? 'translate-x-6' : ''
  191. }`}
  192. />
  193. </button>
  194. </div>
  195. {/* Printer Model - only for non-proxy modes */}
  196. {localMode !== 'proxy' && (
  197. <div className="py-3 border-t border-bambu-dark-tertiary">
  198. <div className="text-white font-medium mb-2">{t('virtualPrinter.model.title')}</div>
  199. <div className="text-sm text-bambu-gray mb-3">
  200. {t('virtualPrinter.model.description')}
  201. </div>
  202. <div className="relative">
  203. <select
  204. value={localModel}
  205. onChange={(e) => handleModelChange(e.target.value)}
  206. disabled={pendingAction === 'model'}
  207. className="w-full bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-md px-3 py-2 text-white appearance-none cursor-pointer disabled:opacity-50 disabled:cursor-not-allowed pr-10"
  208. >
  209. {modelsData?.models && Object.entries(modelsData.models)
  210. .sort(([, a], [, b]) => (a as string).localeCompare(b as string))
  211. .map(([code, name]) => (
  212. <option key={code} value={code}>
  213. {name}
  214. </option>
  215. ))}
  216. </select>
  217. <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
  218. </div>
  219. {localEnabled && isRunning && (
  220. <p className="text-xs text-bambu-gray mt-2">
  221. <Info className="w-3 h-3 inline mr-1" />
  222. {t('virtualPrinter.model.restartWarning')}
  223. </p>
  224. )}
  225. </div>
  226. )}
  227. {/* Access Code - only for non-proxy modes */}
  228. {localMode !== 'proxy' && (
  229. <div className="py-3 border-t border-bambu-dark-tertiary">
  230. <div className="text-white font-medium mb-2">{t('virtualPrinter.accessCode.title')}</div>
  231. <div className="text-sm text-bambu-gray mb-3">
  232. {settings?.access_code_set ? (
  233. <span className="flex items-center gap-1 text-green-400">
  234. <Check className="w-4 h-4" />
  235. {t('virtualPrinter.accessCode.isSet')}
  236. </span>
  237. ) : (
  238. <span className="flex items-center gap-1 text-yellow-400">
  239. <AlertTriangle className="w-4 h-4" />
  240. {t('virtualPrinter.accessCode.notSet')}
  241. </span>
  242. )}
  243. </div>
  244. <div className="flex gap-2">
  245. <div className="relative flex-1">
  246. <input
  247. type={showAccessCode ? 'text' : 'password'}
  248. value={localAccessCode}
  249. onChange={(e) => setLocalAccessCode(e.target.value)}
  250. placeholder={settings?.access_code_set ? t('virtualPrinter.accessCode.placeholderChange') : t('virtualPrinter.accessCode.placeholder')}
  251. maxLength={8}
  252. className="w-full bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-md px-3 py-2 text-white placeholder-bambu-gray pr-10 font-mono"
  253. />
  254. <button
  255. onClick={() => setShowAccessCode(!showAccessCode)}
  256. className="absolute right-2 top-1/2 -translate-y-1/2 text-bambu-gray hover:text-white"
  257. >
  258. {showAccessCode ? <EyeOff className="w-4 h-4" /> : <Eye className="w-4 h-4" />}
  259. </button>
  260. </div>
  261. <Button
  262. onClick={handleAccessCodeChange}
  263. disabled={!localAccessCode || pendingAction === 'accessCode'}
  264. variant="primary"
  265. >
  266. {pendingAction === 'accessCode' ? <Loader2 className="w-4 h-4 animate-spin" /> : t('common.save')}
  267. </Button>
  268. </div>
  269. <p className="text-xs text-bambu-gray mt-2">
  270. {t('virtualPrinter.accessCode.hint')}
  271. {localAccessCode && (
  272. <span className={localAccessCode.length === 8 ? 'text-green-400' : 'text-yellow-400'}>
  273. {' '}{t('virtualPrinter.accessCode.charCount', { count: localAccessCode.length })}
  274. </span>
  275. )}
  276. </p>
  277. </div>
  278. )}
  279. {/* Target Printer - only for proxy mode */}
  280. {localMode === 'proxy' && (
  281. <div className="py-3 border-t border-bambu-dark-tertiary">
  282. <div className="text-white font-medium mb-2">{t('virtualPrinter.targetPrinter.title')}</div>
  283. <div className="text-sm text-bambu-gray mb-3">
  284. {localTargetPrinterId ? (
  285. <span className="flex items-center gap-1 text-green-400">
  286. <Check className="w-4 h-4" />
  287. {t('virtualPrinter.targetPrinter.configured')}
  288. </span>
  289. ) : (
  290. <span className="flex items-center gap-1 text-yellow-400">
  291. <AlertTriangle className="w-4 h-4" />
  292. {t('virtualPrinter.targetPrinter.notConfigured')}
  293. </span>
  294. )}
  295. </div>
  296. <div className="relative">
  297. <select
  298. value={localTargetPrinterId ?? ''}
  299. onChange={(e) => {
  300. const id = parseInt(e.target.value, 10);
  301. if (!isNaN(id)) {
  302. handleTargetPrinterChange(id);
  303. }
  304. }}
  305. disabled={pendingAction === 'targetPrinter'}
  306. className="w-full bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-md px-3 py-2 text-white appearance-none cursor-pointer disabled:opacity-50 disabled:cursor-not-allowed pr-10"
  307. >
  308. <option value="">{t('virtualPrinter.targetPrinter.placeholder')}</option>
  309. {printers?.map((printer) => (
  310. <option key={printer.id} value={printer.id}>
  311. {printer.name} ({printer.ip_address})
  312. </option>
  313. ))}
  314. </select>
  315. <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
  316. </div>
  317. <p className="text-xs text-bambu-gray mt-2">
  318. {t('virtualPrinter.targetPrinter.hint')}
  319. </p>
  320. {!printers?.length && (
  321. <p className="text-xs text-yellow-400 mt-2">
  322. <AlertTriangle className="w-3 h-3 inline mr-1" />
  323. {t('virtualPrinter.targetPrinter.noPrinters')}
  324. </p>
  325. )}
  326. </div>
  327. )}
  328. {/* Mode */}
  329. <div className="py-3 border-t border-bambu-dark-tertiary">
  330. <div className="text-white font-medium mb-2">{t('virtualPrinter.mode.title')}</div>
  331. <div className="grid grid-cols-2 gap-3">
  332. <button
  333. onClick={() => handleModeChange('immediate')}
  334. disabled={pendingAction === 'mode'}
  335. className={`p-3 rounded-lg border text-left transition-colors ${
  336. localMode === 'immediate'
  337. ? 'border-bambu-green bg-bambu-green/10'
  338. : 'border-bambu-dark-tertiary hover:border-bambu-gray'
  339. }`}
  340. >
  341. <div className="text-white font-medium">{t('virtualPrinter.mode.archive')}</div>
  342. <div className="text-xs text-bambu-gray">{t('virtualPrinter.mode.archiveDesc')}</div>
  343. </button>
  344. <button
  345. onClick={() => handleModeChange('review')}
  346. disabled={pendingAction === 'mode'}
  347. className={`p-3 rounded-lg border text-left transition-colors ${
  348. localMode === 'review'
  349. ? 'border-bambu-green bg-bambu-green/10'
  350. : 'border-bambu-dark-tertiary hover:border-bambu-gray'
  351. }`}
  352. >
  353. <div className="text-white font-medium">{t('virtualPrinter.mode.review')}</div>
  354. <div className="text-xs text-bambu-gray">{t('virtualPrinter.mode.reviewDesc')}</div>
  355. </button>
  356. <button
  357. onClick={() => handleModeChange('print_queue')}
  358. disabled={pendingAction === 'mode'}
  359. className={`p-3 rounded-lg border text-left transition-colors ${
  360. localMode === 'print_queue'
  361. ? 'border-bambu-green bg-bambu-green/10'
  362. : 'border-bambu-dark-tertiary hover:border-bambu-gray'
  363. }`}
  364. >
  365. <div className="text-white font-medium">{t('virtualPrinter.mode.queue')}</div>
  366. <div className="text-xs text-bambu-gray">{t('virtualPrinter.mode.queueDesc')}</div>
  367. </button>
  368. <button
  369. onClick={() => handleModeChange('proxy')}
  370. disabled={pendingAction === 'mode'}
  371. className={`p-3 rounded-lg border text-left transition-colors ${
  372. localMode === 'proxy'
  373. ? 'border-blue-500 bg-blue-500/10'
  374. : 'border-bambu-dark-tertiary hover:border-bambu-gray'
  375. }`}
  376. >
  377. <div className="flex items-center gap-1.5 text-white font-medium">
  378. <ArrowRightLeft className="w-4 h-4" />
  379. {t('virtualPrinter.mode.proxy')}
  380. </div>
  381. <div className="text-xs text-bambu-gray">{t('virtualPrinter.mode.proxyDesc')}</div>
  382. </button>
  383. </div>
  384. </div>
  385. </CardContent>
  386. </Card>
  387. </div>
  388. {/* Right Column - Info & Status */}
  389. <div className="space-y-6 lg:w-[480px] lg:flex-shrink-0">
  390. {/* Setup Required Warning */}
  391. <Card className="border-l-4 border-l-yellow-500">
  392. <CardContent className="py-4">
  393. <div className="flex items-start gap-3">
  394. <AlertTriangle className="w-5 h-5 text-yellow-500 flex-shrink-0 mt-0.5" />
  395. <div className="text-sm">
  396. <p className="text-white font-medium mb-2">
  397. {t('virtualPrinter.setupRequired.title')}
  398. </p>
  399. <p className="text-bambu-gray mb-3">
  400. {t('virtualPrinter.setupRequired.description')}
  401. </p>
  402. <a
  403. href="https://wiki.bambuddy.cool/features/virtual-printer/"
  404. target="_blank"
  405. rel="noopener noreferrer"
  406. className="inline-flex items-center gap-2 px-4 py-2 bg-yellow-500/20 border border-yellow-500/50 rounded-md text-yellow-400 hover:bg-yellow-500/30 transition-colors"
  407. >
  408. <ExternalLink className="w-4 h-4" />
  409. {t('virtualPrinter.setupRequired.readGuide')}
  410. </a>
  411. </div>
  412. </div>
  413. </CardContent>
  414. </Card>
  415. {/* How it works */}
  416. <Card>
  417. <CardContent className="py-4">
  418. <div className="flex items-start gap-3">
  419. <Info className="w-5 h-5 text-blue-400 flex-shrink-0 mt-0.5" />
  420. <div className="text-sm text-bambu-gray">
  421. <p className="mb-2">
  422. <strong className="text-white">{localMode === 'proxy' ? t('virtualPrinter.howItWorks.titleProxy') : t('virtualPrinter.howItWorks.title')}:</strong>
  423. </p>
  424. {localMode === 'proxy' ? (
  425. <ol className="list-decimal list-inside space-y-1">
  426. <li>{t('virtualPrinter.howItWorks.proxyStep1')}</li>
  427. <li>{t('virtualPrinter.howItWorks.proxyStep2')}</li>
  428. <li>{t('virtualPrinter.howItWorks.proxyStep3')}</li>
  429. <li>{t('virtualPrinter.howItWorks.proxyStep4')}</li>
  430. <li>{t('virtualPrinter.howItWorks.proxyStep5')}</li>
  431. </ol>
  432. ) : (
  433. <ol className="list-decimal list-inside space-y-1">
  434. <li>{t('virtualPrinter.howItWorks.step1')}</li>
  435. <li>{t('virtualPrinter.howItWorks.step2')}</li>
  436. <li>{t('virtualPrinter.howItWorks.step3')}</li>
  437. <li>{t('virtualPrinter.howItWorks.step4')}</li>
  438. <li>{t('virtualPrinter.howItWorks.step5')}</li>
  439. <li>{t('virtualPrinter.howItWorks.step6')}</li>
  440. </ol>
  441. )}
  442. </div>
  443. </div>
  444. </CardContent>
  445. </Card>
  446. {/* Status Details (when running) */}
  447. {status && isRunning && (
  448. <Card>
  449. <CardHeader>
  450. <h3 className="text-md font-semibold text-white">{t('virtualPrinter.status.title')}</h3>
  451. </CardHeader>
  452. <CardContent>
  453. {status.mode === 'proxy' && status.proxy ? (
  454. <div className="grid grid-cols-2 gap-4 text-sm">
  455. <div>
  456. <div className="text-bambu-gray">{t('virtualPrinter.status.targetPrinter')}</div>
  457. <div className="text-white">
  458. {printers?.find(p => p.id === localTargetPrinterId)?.name || status.proxy.target_host}
  459. </div>
  460. <div className="text-xs text-bambu-gray font-mono">{status.proxy.target_host}</div>
  461. </div>
  462. <div>
  463. <div className="text-bambu-gray">{t('virtualPrinter.status.mode')}</div>
  464. <div className="text-white flex items-center gap-1.5">
  465. <ArrowRightLeft className="w-4 h-4" />
  466. {t('virtualPrinter.mode.proxy')}
  467. </div>
  468. </div>
  469. <div>
  470. <div className="text-bambu-gray">{t('virtualPrinter.status.ftpPort')}</div>
  471. <div className="text-white font-mono">{status.proxy.ftp_port}</div>
  472. </div>
  473. <div>
  474. <div className="text-bambu-gray">{t('virtualPrinter.status.mqttPort')}</div>
  475. <div className="text-white font-mono">{status.proxy.mqtt_port}</div>
  476. </div>
  477. <div>
  478. <div className="text-bambu-gray">{t('virtualPrinter.status.ftpConnections')}</div>
  479. <div className="text-white">{status.proxy.ftp_connections}</div>
  480. </div>
  481. <div>
  482. <div className="text-bambu-gray">{t('virtualPrinter.status.mqttConnections')}</div>
  483. <div className="text-white">{status.proxy.mqtt_connections}</div>
  484. </div>
  485. </div>
  486. ) : (
  487. <div className="grid grid-cols-2 gap-4 text-sm">
  488. <div>
  489. <div className="text-bambu-gray">{t('virtualPrinter.status.printerName')}</div>
  490. <div className="text-white">{status.name}</div>
  491. </div>
  492. <div>
  493. <div className="text-bambu-gray">{t('virtualPrinter.status.model')}</div>
  494. <div className="text-white">{status.model_name || status.model}</div>
  495. </div>
  496. <div>
  497. <div className="text-bambu-gray">{t('virtualPrinter.status.serialNumber')}</div>
  498. <div className="text-white font-mono">{status.serial}</div>
  499. </div>
  500. <div>
  501. <div className="text-bambu-gray">{t('virtualPrinter.status.mode')}</div>
  502. <div className="text-white capitalize">{status.mode}</div>
  503. </div>
  504. <div>
  505. <div className="text-bambu-gray">{t('virtualPrinter.status.pendingFiles')}</div>
  506. <div className="text-white">{status.pending_files}</div>
  507. </div>
  508. </div>
  509. )}
  510. </CardContent>
  511. </Card>
  512. )}
  513. </div>
  514. </div>
  515. );
  516. }