VirtualPrinterSettings.tsx 27 KB

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