PrinterInfoModal.tsx 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. import { useState, useEffect } from 'react';
  2. import { useTranslation } from 'react-i18next';
  3. import { X, Copy, Check, Signal } from 'lucide-react';
  4. import { Card, CardContent } from './Card';
  5. import { formatDateOnly } from '../utils/date';
  6. import { getPrinterImage, getWifiStrength } from '../utils/printer';
  7. import type { Printer, PrinterStatus } from '../api/client';
  8. interface PrinterInfoModalProps {
  9. printer: Printer;
  10. status?: PrinterStatus;
  11. totalPrintHours?: number;
  12. onClose: () => void;
  13. }
  14. function CopyButton({ value }: { value: string }) {
  15. const { t } = useTranslation();
  16. const [copied, setCopied] = useState(false);
  17. const handleCopy = async () => {
  18. try {
  19. await navigator.clipboard.writeText(value);
  20. setCopied(true);
  21. setTimeout(() => setCopied(false), 2000);
  22. } catch {
  23. // Clipboard may not be available in non-secure contexts
  24. }
  25. };
  26. return (
  27. <button
  28. onClick={handleCopy}
  29. className="ml-2 p-1 rounded hover:bg-bambu-dark-tertiary text-bambu-gray hover:text-white transition-colors"
  30. title={copied ? t('printers.copied') : t('printers.copyToClipboard')}
  31. >
  32. {copied ? <Check className="w-3.5 h-3.5 text-bambu-green" /> : <Copy className="w-3.5 h-3.5" />}
  33. </button>
  34. );
  35. }
  36. export function PrinterInfoModal({ printer, status, totalPrintHours, onClose }: PrinterInfoModalProps) {
  37. const { t } = useTranslation();
  38. useEffect(() => {
  39. const handleKey = (e: KeyboardEvent) => {
  40. if (e.key === 'Escape') onClose();
  41. };
  42. window.addEventListener('keydown', handleKey);
  43. return () => window.removeEventListener('keydown', handleKey);
  44. }, [onClose]);
  45. const rows: { label: string; value: React.ReactNode }[] = [];
  46. // Model
  47. rows.push({
  48. label: t('printers.model'),
  49. value: printer.model ?? '—',
  50. });
  51. // Connection Status
  52. rows.push({
  53. label: t('common.status'),
  54. value: (
  55. <span className={`inline-flex items-center gap-1.5 px-2 py-0.5 rounded-full text-xs font-medium ${
  56. status?.connected
  57. ? 'bg-bambu-green/20 text-bambu-green'
  58. : 'bg-red-500/20 text-red-400'
  59. }`}>
  60. <span className={`w-1.5 h-1.5 rounded-full ${status?.connected ? 'bg-bambu-green' : 'bg-red-400'}`} />
  61. {status?.connected ? t('printers.status.available') : t('printers.status.offline')}
  62. </span>
  63. ),
  64. });
  65. // State
  66. if (status?.state) {
  67. const stateMap: Record<string, string> = {
  68. IDLE: 'printers.status.idle',
  69. RUNNING: 'printers.status.printing',
  70. PAUSE: 'printers.status.paused',
  71. FINISH: 'printers.status.finished',
  72. FAILED: 'printers.status.error',
  73. };
  74. rows.push({
  75. label: t('printers.state'),
  76. value: t(stateMap[status.state] ?? 'printers.status.unknown'),
  77. });
  78. }
  79. // IP Address
  80. rows.push({
  81. label: t('printers.ipAddress'),
  82. value: (
  83. <span className="flex items-center">
  84. <span className="font-mono">{printer.ip_address}</span>
  85. <CopyButton value={printer.ip_address} />
  86. </span>
  87. ),
  88. });
  89. // Serial Number
  90. rows.push({
  91. label: t('printers.serialNumber'),
  92. value: (
  93. <span className="flex items-center">
  94. <span className="font-mono truncate">{printer.serial_number}</span>
  95. <CopyButton value={printer.serial_number} />
  96. </span>
  97. ),
  98. });
  99. // WiFi Signal
  100. if (status?.wifi_signal != null) {
  101. const wifi = getWifiStrength(status.wifi_signal);
  102. rows.push({
  103. label: t('printers.wifiSignalLabel'),
  104. value: (
  105. <span className="flex items-center gap-2">
  106. <Signal className={`w-4 h-4 ${wifi.color}`} />
  107. <span className={wifi.color}>{t(wifi.labelKey)}</span>
  108. <span className="text-bambu-gray text-xs">({status.wifi_signal} dBm)</span>
  109. </span>
  110. ),
  111. });
  112. }
  113. // Firmware
  114. rows.push({
  115. label: t('printers.firmware'),
  116. value: status?.firmware_version ?? '—',
  117. });
  118. // Developer Mode
  119. if (status?.developer_mode != null) {
  120. rows.push({
  121. label: t('printers.developerMode'),
  122. value: (
  123. <span className={`inline-flex items-center px-2 py-0.5 rounded-full text-xs font-medium ${
  124. status.developer_mode
  125. ? 'bg-bambu-green/20 text-bambu-green'
  126. : 'bg-bambu-dark-tertiary text-bambu-gray'
  127. }`}>
  128. {status.developer_mode ? t('printers.enabled') : t('printers.disabled')}
  129. </span>
  130. ),
  131. });
  132. }
  133. // Nozzle Count
  134. rows.push({
  135. label: t('printers.nozzleCount'),
  136. value: printer.nozzle_count,
  137. });
  138. // SD Card
  139. if (status?.sdcard != null) {
  140. rows.push({
  141. label: t('printers.sdCard'),
  142. value: (
  143. <span className={`inline-flex items-center px-2 py-0.5 rounded-full text-xs font-medium ${
  144. status.sdcard
  145. ? 'bg-bambu-green/20 text-bambu-green'
  146. : 'bg-bambu-dark-tertiary text-bambu-gray'
  147. }`}>
  148. {status.sdcard ? t('printers.inserted') : t('printers.notInserted')}
  149. </span>
  150. ),
  151. });
  152. }
  153. // Auto-Archive
  154. rows.push({
  155. label: t('printers.autoArchive'),
  156. value: (
  157. <span className={`inline-flex items-center px-2 py-0.5 rounded-full text-xs font-medium ${
  158. printer.auto_archive
  159. ? 'bg-bambu-green/20 text-bambu-green'
  160. : 'bg-bambu-dark-tertiary text-bambu-gray'
  161. }`}>
  162. {printer.auto_archive ? t('printers.enabled') : t('printers.disabled')}
  163. </span>
  164. ),
  165. });
  166. // Total Print Hours
  167. if (totalPrintHours != null && totalPrintHours > 0) {
  168. rows.push({
  169. label: t('printers.totalPrintHours'),
  170. value: `${Math.round(totalPrintHours)}h`,
  171. });
  172. }
  173. // Location
  174. if (printer.location) {
  175. rows.push({
  176. label: t('printers.sort.location'),
  177. value: printer.location,
  178. });
  179. }
  180. // Added date
  181. rows.push({
  182. label: t('printers.addedOn'),
  183. value: formatDateOnly(printer.created_at),
  184. });
  185. return (
  186. <div
  187. className="fixed inset-0 bg-black/50 flex items-center justify-center z-50 p-4"
  188. role="dialog"
  189. aria-modal="true"
  190. onClick={onClose}
  191. >
  192. <Card className="w-full max-w-md" onClick={(e: React.MouseEvent) => e.stopPropagation()}>
  193. <CardContent>
  194. <div className="flex items-center justify-between mb-4">
  195. <h2 className="text-lg font-semibold text-white">
  196. {printer.name}
  197. </h2>
  198. <button onClick={onClose} className="p-1 hover:bg-bambu-dark rounded flex-shrink-0">
  199. <X className="w-5 h-5 text-bambu-gray" />
  200. </button>
  201. </div>
  202. {/* Printer Image */}
  203. <div className="flex justify-center mb-4">
  204. <img
  205. src={getPrinterImage(printer.model)}
  206. alt={printer.model ?? printer.name}
  207. className="h-24 object-contain"
  208. />
  209. </div>
  210. <div className="space-y-0">
  211. {rows.map((row, i) => (
  212. <div key={i} className="flex items-center justify-between gap-4 py-2.5 border-b border-bambu-dark-tertiary last:border-0">
  213. <span className="text-sm text-bambu-gray whitespace-nowrap">{row.label}</span>
  214. <span className="text-sm text-white text-right">{row.value}</span>
  215. </div>
  216. ))}
  217. </div>
  218. </CardContent>
  219. </Card>
  220. </div>
  221. );
  222. }