AddNotificationModal.tsx 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559
  1. import { useState, useEffect } from 'react';
  2. import { useMutation, useQueryClient, useQuery } from '@tanstack/react-query';
  3. import { useTranslation } from 'react-i18next';
  4. import { X, Save, Loader2, Send, CheckCircle, XCircle } from 'lucide-react';
  5. import { api } from '../api/client';
  6. import type { NotificationProvider, NotificationProviderCreate, NotificationProviderUpdate, ProviderType } from '../api/client';
  7. import { Button } from './Button';
  8. import { Toggle } from './Toggle';
  9. interface AddNotificationModalProps {
  10. provider?: NotificationProvider | null;
  11. onClose: () => void;
  12. }
  13. const PROVIDER_VALUES: ProviderType[] = ['email', 'telegram', 'discord', 'ntfy', 'pushover', 'callmebot', 'webhook', 'homeassistant'];
  14. export function AddNotificationModal({ provider, onClose }: AddNotificationModalProps) {
  15. const { t } = useTranslation();
  16. const queryClient = useQueryClient();
  17. const isEditing = !!provider;
  18. const [name, setName] = useState(provider?.name || '');
  19. const [providerType, setProviderType] = useState<ProviderType>(provider?.provider_type || 'email');
  20. const [printerId, setPrinterId] = useState<number | null>(provider?.printer_id || null);
  21. const [quietHoursEnabled, setQuietHoursEnabled] = useState(provider?.quiet_hours_enabled || false);
  22. const [quietHoursStart, setQuietHoursStart] = useState(provider?.quiet_hours_start || '22:00');
  23. const [quietHoursEnd, setQuietHoursEnd] = useState(provider?.quiet_hours_end || '07:00');
  24. // Daily digest
  25. const [dailyDigestEnabled, setDailyDigestEnabled] = useState(provider?.daily_digest_enabled || false);
  26. const [dailyDigestTime, setDailyDigestTime] = useState(provider?.daily_digest_time || '08:00');
  27. // Event toggles
  28. const [onPrintStart, setOnPrintStart] = useState(provider?.on_print_start ?? false);
  29. const [onPrintComplete, setOnPrintComplete] = useState(provider?.on_print_complete ?? true);
  30. const [onPrintFailed, setOnPrintFailed] = useState(provider?.on_print_failed ?? true);
  31. const [onPrintStopped, setOnPrintStopped] = useState(provider?.on_print_stopped ?? true);
  32. const [onPrintProgress, setOnPrintProgress] = useState(provider?.on_print_progress ?? false);
  33. const [onPrinterOffline, setOnPrinterOffline] = useState(provider?.on_printer_offline ?? false);
  34. const [onPrinterError, setOnPrinterError] = useState(provider?.on_printer_error ?? false);
  35. const [onFilamentLow, setOnFilamentLow] = useState(provider?.on_filament_low ?? false);
  36. const [onMaintenanceDue, setOnMaintenanceDue] = useState(provider?.on_maintenance_due ?? false);
  37. const [onBedCooled, setOnBedCooled] = useState(provider?.on_bed_cooled ?? false);
  38. const [onFirstLayerComplete, setOnFirstLayerComplete] = useState(provider?.on_first_layer_complete ?? false);
  39. // Provider-specific config
  40. const [config, setConfig] = useState<Record<string, string>>(
  41. provider?.config ? Object.fromEntries(Object.entries(provider.config).map(([k, v]) => [k, String(v)])) : {}
  42. );
  43. const [testResult, setTestResult] = useState<{ success: boolean; message: string } | null>(null);
  44. const [error, setError] = useState<string | null>(null);
  45. // Fetch printers for linking
  46. const { data: printers } = useQuery({
  47. queryKey: ['printers'],
  48. queryFn: api.getPrinters,
  49. });
  50. // Close on Escape key
  51. useEffect(() => {
  52. const handleKeyDown = (e: KeyboardEvent) => {
  53. if (e.key === 'Escape') onClose();
  54. };
  55. window.addEventListener('keydown', handleKeyDown);
  56. return () => window.removeEventListener('keydown', handleKeyDown);
  57. }, [onClose]);
  58. // Test configuration mutation
  59. const testMutation = useMutation({
  60. mutationFn: () => api.testNotificationConfig({ provider_type: providerType, config }),
  61. onSuccess: (result) => {
  62. setTestResult(result);
  63. setError(null);
  64. },
  65. onError: (err: Error) => {
  66. setTestResult({ success: false, message: err.message });
  67. },
  68. });
  69. // Create mutation
  70. const createMutation = useMutation({
  71. mutationFn: (data: NotificationProviderCreate) => api.createNotificationProvider(data),
  72. onSuccess: () => {
  73. queryClient.invalidateQueries({ queryKey: ['notification-providers'] });
  74. onClose();
  75. },
  76. onError: (err: Error) => {
  77. setError(err.message);
  78. },
  79. });
  80. // Update mutation
  81. const updateMutation = useMutation({
  82. mutationFn: (data: NotificationProviderUpdate) => api.updateNotificationProvider(provider!.id, data),
  83. onSuccess: () => {
  84. queryClient.invalidateQueries({ queryKey: ['notification-providers'] });
  85. onClose();
  86. },
  87. onError: (err: Error) => {
  88. setError(err.message);
  89. },
  90. });
  91. const handleSubmit = (e: React.FormEvent) => {
  92. e.preventDefault();
  93. setError(null);
  94. if (!name.trim()) {
  95. setError(t('notifications.nameRequired'));
  96. return;
  97. }
  98. // Validate provider-specific config
  99. const requiredFields = getRequiredFields(providerType);
  100. for (const field of requiredFields) {
  101. if (!config[field.key]?.trim()) {
  102. setError(t('notifications.fieldRequired', { field: field.label }));
  103. return;
  104. }
  105. }
  106. const data = {
  107. name: name.trim(),
  108. provider_type: providerType,
  109. config,
  110. printer_id: printerId,
  111. quiet_hours_enabled: quietHoursEnabled,
  112. quiet_hours_start: quietHoursEnabled ? quietHoursStart : null,
  113. quiet_hours_end: quietHoursEnabled ? quietHoursEnd : null,
  114. // Daily digest
  115. daily_digest_enabled: dailyDigestEnabled,
  116. daily_digest_time: dailyDigestEnabled ? dailyDigestTime : null,
  117. // Event toggles
  118. on_print_start: onPrintStart,
  119. on_print_complete: onPrintComplete,
  120. on_print_failed: onPrintFailed,
  121. on_print_stopped: onPrintStopped,
  122. on_print_progress: onPrintProgress,
  123. on_printer_offline: onPrinterOffline,
  124. on_printer_error: onPrinterError,
  125. on_filament_low: onFilamentLow,
  126. on_maintenance_due: onMaintenanceDue,
  127. on_bed_cooled: onBedCooled,
  128. on_first_layer_complete: onFirstLayerComplete,
  129. };
  130. if (isEditing) {
  131. updateMutation.mutate(data);
  132. } else {
  133. createMutation.mutate(data);
  134. }
  135. };
  136. const isPending = createMutation.isPending || updateMutation.isPending;
  137. // Get config fields for each provider type
  138. const getConfigFields = (type: ProviderType) => {
  139. switch (type) {
  140. case 'callmebot':
  141. return [
  142. { key: 'phone', label: 'Phone Number', placeholder: '+1234567890', type: 'text', required: true },
  143. { key: 'apikey', label: 'API Key', placeholder: 'Your CallMeBot API key', type: 'text', required: true },
  144. ];
  145. case 'ntfy':
  146. return [
  147. { key: 'server', label: 'Server URL', placeholder: 'https://ntfy.sh', type: 'text', required: false },
  148. { key: 'topic', label: 'Topic', placeholder: 'my-bambuddy', type: 'text', required: true },
  149. { key: 'auth_token', label: 'Auth Token', placeholder: 'Optional authentication', type: 'password', required: false },
  150. ];
  151. case 'pushover':
  152. return [
  153. { key: 'user_key', label: 'User Key', placeholder: 'Your Pushover user key', type: 'text', required: true },
  154. { key: 'app_token', label: 'App Token', placeholder: 'Your Pushover app token', type: 'text', required: true },
  155. { key: 'priority', label: 'Priority', placeholder: '0 (normal)', type: 'number', required: false },
  156. ];
  157. case 'telegram':
  158. return [
  159. { key: 'bot_token', label: 'Bot Token', placeholder: 'Bot token from @BotFather', type: 'password', required: true },
  160. { key: 'chat_id', label: 'Chat ID', placeholder: 'Your chat or group ID', type: 'text', required: true },
  161. ];
  162. case 'email':
  163. return [
  164. { key: 'smtp_server', label: 'SMTP Server', placeholder: 'smtp.gmail.com', type: 'text', required: true },
  165. { key: 'smtp_port', label: 'SMTP Port', placeholder: '587', type: 'number', required: false },
  166. { key: 'security', label: 'Security', type: 'select', required: false, options: [
  167. { value: 'starttls', label: 'STARTTLS (Port 587)' },
  168. { value: 'ssl', label: 'SSL/TLS (Port 465)' },
  169. { value: 'none', label: 'None (Port 25)' },
  170. ]},
  171. { key: 'auth_enabled', label: 'Authentication', type: 'select', required: false, options: [
  172. { value: 'true', label: 'Enabled' },
  173. { value: 'false', label: 'Disabled' },
  174. ]},
  175. { key: 'username', label: 'Username', placeholder: 'your@email.com', type: 'text', required: false },
  176. { key: 'password', label: 'Password', placeholder: 'App password', type: 'password', required: false },
  177. { key: 'from_email', label: 'From Email', placeholder: 'your@email.com', type: 'text', required: true },
  178. { key: 'to_email', label: 'To Email', placeholder: 'recipient@email.com', type: 'text', required: true },
  179. ];
  180. case 'discord':
  181. return [
  182. { key: 'webhook_url', label: 'Webhook URL', placeholder: 'https://discord.com/api/webhooks/...', type: 'text', required: true },
  183. ];
  184. case 'webhook':
  185. return [
  186. { key: 'webhook_url', label: 'Webhook URL', placeholder: 'https://example.com/webhook', type: 'text', required: true },
  187. { key: 'payload_format', label: 'Payload Format', type: 'select', required: false, options: [
  188. { value: 'generic', label: 'Generic JSON' },
  189. { value: 'slack', label: 'Slack / Mattermost' },
  190. ]},
  191. { key: 'auth_header', label: 'Authorization', placeholder: 'Bearer token (optional)', type: 'password', required: false },
  192. { key: 'field_title', label: 'Title Field Name', placeholder: 'title', type: 'text', required: false, showIf: (cfg: Record<string, string>) => cfg.payload_format !== 'slack' },
  193. { key: 'field_message', label: 'Message Field Name', placeholder: 'message', type: 'text', required: false, showIf: (cfg: Record<string, string>) => cfg.payload_format !== 'slack' },
  194. ];
  195. case 'homeassistant':
  196. return [
  197. { key: 'service', label: 'Home Assistant Service', placeholder: 'notify.mobile_app_myphone', type: 'text', required: false },
  198. ];
  199. default:
  200. return [];
  201. }
  202. };
  203. const getRequiredFields = (type: ProviderType) => {
  204. return getConfigFields(type).filter(f => f.required);
  205. };
  206. const configFields = getConfigFields(providerType);
  207. return (
  208. <div
  209. className="fixed inset-0 bg-black/70 flex items-center justify-center z-50 p-4 overflow-y-auto"
  210. onClick={onClose}
  211. >
  212. <div
  213. className="bg-bambu-dark-secondary rounded-xl border border-bambu-dark-tertiary w-full max-w-lg my-8 max-h-[90vh] overflow-y-auto"
  214. onClick={(e) => e.stopPropagation()}
  215. >
  216. {/* Header */}
  217. <div className="flex items-center justify-between px-6 py-4 border-b border-bambu-dark-tertiary">
  218. <h2 className="text-lg font-semibold text-white">
  219. {isEditing ? t('notifications.editTitle') : t('notifications.addTitle')}
  220. </h2>
  221. <button
  222. onClick={onClose}
  223. className="text-bambu-gray hover:text-white transition-colors"
  224. >
  225. <X className="w-5 h-5" />
  226. </button>
  227. </div>
  228. {/* Form */}
  229. <form onSubmit={handleSubmit} className="p-6 space-y-4">
  230. {error && (
  231. <div className="p-3 bg-red-500/20 border border-red-500/50 rounded-lg text-sm text-red-400">
  232. {error}
  233. </div>
  234. )}
  235. {/* Name */}
  236. <div>
  237. <label className="block text-sm text-bambu-gray mb-1">{t('notifications.nameLabel')}</label>
  238. <input
  239. type="text"
  240. value={name}
  241. onChange={(e) => setName(e.target.value)}
  242. placeholder={t('notifications.namePlaceholder')}
  243. className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  244. />
  245. </div>
  246. {/* Provider Type */}
  247. <div>
  248. <label className="block text-sm text-bambu-gray mb-1">{t('notifications.providerTypeLabel')}</label>
  249. <select
  250. value={providerType}
  251. onChange={(e) => {
  252. setProviderType(e.target.value as ProviderType);
  253. setConfig({}); // Reset config when changing type
  254. setTestResult(null);
  255. }}
  256. disabled={isEditing}
  257. className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none disabled:opacity-50"
  258. >
  259. {PROVIDER_VALUES.map((value) => (
  260. <option key={value} value={value}>
  261. {t(`notifications.providerTypes.${value}`, value)}
  262. </option>
  263. ))}
  264. </select>
  265. <p className="text-xs text-bambu-gray mt-1">
  266. {t(`notifications.providerDescriptions.${providerType}`, '')}
  267. </p>
  268. </div>
  269. {/* Provider-specific configuration */}
  270. <div className="space-y-3">
  271. <p className="text-sm text-bambu-gray">{t('notifications.configuration')}</p>
  272. {configFields
  273. .filter((field) => !('showIf' in field) || (field as { showIf?: (cfg: Record<string, string>) => boolean }).showIf?.(config) !== false)
  274. .map((field) => (
  275. <div key={field.key}>
  276. <label className="block text-sm text-bambu-gray mb-1">
  277. {field.label} {field.required && '*'}
  278. </label>
  279. {field.type === 'select' && 'options' in field && field.options ? (
  280. <select
  281. value={config[field.key] || field.options[0]?.value || ''}
  282. onChange={(e) => {
  283. setConfig({ ...config, [field.key]: e.target.value });
  284. setTestResult(null);
  285. }}
  286. className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  287. >
  288. {field.options.map((opt) => (
  289. <option key={opt.value} value={opt.value}>
  290. {opt.label}
  291. </option>
  292. ))}
  293. </select>
  294. ) : (
  295. <input
  296. type={field.type}
  297. value={config[field.key] || ''}
  298. onChange={(e) => {
  299. setConfig({ ...config, [field.key]: e.target.value });
  300. setTestResult(null);
  301. }}
  302. placeholder={field.placeholder}
  303. className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  304. />
  305. )}
  306. </div>
  307. ))}
  308. </div>
  309. {/* Test Button */}
  310. <div className="flex gap-2">
  311. <Button
  312. type="button"
  313. variant="secondary"
  314. onClick={() => {
  315. setTestResult(null);
  316. testMutation.mutate();
  317. }}
  318. disabled={testMutation.isPending || (getRequiredFields(providerType).length > 0 && !config[getRequiredFields(providerType)[0]?.key])}
  319. className="flex-1"
  320. >
  321. {testMutation.isPending ? (
  322. <Loader2 className="w-4 h-4 animate-spin" />
  323. ) : (
  324. <Send className="w-4 h-4" />
  325. )}
  326. {t('notifications.testConfiguration')}
  327. </Button>
  328. </div>
  329. {/* Test Result */}
  330. {testResult && (
  331. <div className={`p-3 rounded-lg flex items-center gap-2 ${
  332. testResult.success
  333. ? 'bg-bambu-green/20 border border-bambu-green/50 text-bambu-green'
  334. : 'bg-red-500/20 border border-red-500/50 text-red-400'
  335. }`}>
  336. {testResult.success ? (
  337. <>
  338. <CheckCircle className="w-5 h-5" />
  339. <span>{testResult.message}</span>
  340. </>
  341. ) : (
  342. <>
  343. <XCircle className="w-5 h-5" />
  344. <span>{testResult.message}</span>
  345. </>
  346. )}
  347. </div>
  348. )}
  349. {/* Link to Printer */}
  350. <div>
  351. <label className="block text-sm text-bambu-gray mb-1">{t('notifications.printerFilter')}</label>
  352. <select
  353. value={printerId ?? ''}
  354. onChange={(e) => setPrinterId(e.target.value ? Number(e.target.value) : null)}
  355. className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  356. >
  357. <option value="">{t('notifications.allPrinters')}</option>
  358. {printers?.map((p) => (
  359. <option key={p.id} value={p.id}>
  360. {p.name}
  361. </option>
  362. ))}
  363. </select>
  364. <p className="text-xs text-bambu-gray mt-1">
  365. {t('notifications.onlyFromPrinter')}
  366. </p>
  367. </div>
  368. {/* Quiet Hours */}
  369. <div className="space-y-2">
  370. <div className="flex items-center justify-between">
  371. <label className="text-sm text-white">{t('notifications.quietHoursDnd')}</label>
  372. <Toggle
  373. checked={quietHoursEnabled}
  374. onChange={setQuietHoursEnabled}
  375. />
  376. </div>
  377. {quietHoursEnabled && (
  378. <div className="grid grid-cols-2 gap-3">
  379. <div>
  380. <label className="block text-xs text-bambu-gray mb-1">{t('notifications.quietStart')}</label>
  381. <input
  382. type="time"
  383. value={quietHoursStart}
  384. onChange={(e) => setQuietHoursStart(e.target.value)}
  385. className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  386. />
  387. </div>
  388. <div>
  389. <label className="block text-xs text-bambu-gray mb-1">{t('notifications.quietEnd')}</label>
  390. <input
  391. type="time"
  392. value={quietHoursEnd}
  393. onChange={(e) => setQuietHoursEnd(e.target.value)}
  394. className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  395. />
  396. </div>
  397. </div>
  398. )}
  399. </div>
  400. {/* Daily Digest */}
  401. <div className="space-y-2">
  402. <div className="flex items-center justify-between">
  403. <div>
  404. <label className="text-sm text-white">{t('notifications.dailyDigestLabel')}</label>
  405. <p className="text-xs text-bambu-gray">{t('notifications.batchNotifications')}</p>
  406. </div>
  407. <Toggle
  408. checked={dailyDigestEnabled}
  409. onChange={setDailyDigestEnabled}
  410. />
  411. </div>
  412. {dailyDigestEnabled && (
  413. <div>
  414. <label className="block text-xs text-bambu-gray mb-1">{t('notifications.sendDigestAt')}</label>
  415. <input
  416. type="time"
  417. value={dailyDigestTime}
  418. onChange={(e) => setDailyDigestTime(e.target.value)}
  419. className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  420. />
  421. <p className="text-xs text-bambu-gray mt-1">
  422. {t('notifications.digestCollected')}
  423. </p>
  424. </div>
  425. )}
  426. </div>
  427. {/* Event Toggles */}
  428. <div className="space-y-3">
  429. <p className="text-sm text-bambu-gray">{t('notifications.notificationEvents')}</p>
  430. {/* Print Events */}
  431. <div className="space-y-2 p-3 bg-bambu-dark rounded-lg">
  432. <p className="text-xs text-bambu-gray uppercase tracking-wide mb-2">{t('notifications.printEvents')}</p>
  433. <div className="grid grid-cols-2 gap-2">
  434. <div className="flex items-center justify-between">
  435. <span className="text-sm text-white">{t('notifications.start')}</span>
  436. <Toggle checked={onPrintStart} onChange={setOnPrintStart} />
  437. </div>
  438. <div className="flex items-center justify-between">
  439. <span className="text-sm text-white">{t('notifications.complete')}</span>
  440. <Toggle checked={onPrintComplete} onChange={setOnPrintComplete} />
  441. </div>
  442. <div className="flex items-center justify-between">
  443. <span className="text-sm text-white">{t('notifications.failed')}</span>
  444. <Toggle checked={onPrintFailed} onChange={setOnPrintFailed} />
  445. </div>
  446. <div className="flex items-center justify-between">
  447. <span className="text-sm text-white">{t('notifications.stopped')}</span>
  448. <Toggle checked={onPrintStopped} onChange={setOnPrintStopped} />
  449. </div>
  450. <div className="flex items-center justify-between col-span-2">
  451. <div>
  452. <span className="text-sm text-white">{t('notifications.progress')}</span>
  453. <span className="text-xs text-bambu-gray ml-1">{t('notifications.progressPercent')}</span>
  454. </div>
  455. <Toggle checked={onPrintProgress} onChange={setOnPrintProgress} />
  456. </div>
  457. <div className="flex items-center justify-between col-span-2">
  458. <div>
  459. <span className="text-sm text-white">{t('notifications.bedCooled')}</span>
  460. <span className="text-xs text-bambu-gray ml-1">{t('notifications.bedCooledAfterPrint')}</span>
  461. </div>
  462. <Toggle checked={onBedCooled} onChange={setOnBedCooled} />
  463. </div>
  464. <div className="flex items-center justify-between col-span-2">
  465. <div>
  466. <span className="text-sm text-white">{t('notifications.firstLayerCompleteLabel')}</span>
  467. <span className="text-xs text-bambu-gray ml-1">{t('notifications.firstLayerCompleteDescription')}</span>
  468. </div>
  469. <Toggle checked={onFirstLayerComplete} onChange={setOnFirstLayerComplete} />
  470. </div>
  471. </div>
  472. </div>
  473. {/* Printer Status Events */}
  474. <div className="space-y-2 p-3 bg-bambu-dark rounded-lg">
  475. <p className="text-xs text-bambu-gray uppercase tracking-wide mb-2">{t('notifications.printerStatus')}</p>
  476. <div className="grid grid-cols-2 gap-2">
  477. <div className="flex items-center justify-between">
  478. <span className="text-sm text-white">{t('notifications.offline')}</span>
  479. <Toggle checked={onPrinterOffline} onChange={setOnPrinterOffline} />
  480. </div>
  481. <div className="flex items-center justify-between">
  482. <span className="text-sm text-white">{t('notifications.error')}</span>
  483. <Toggle checked={onPrinterError} onChange={setOnPrinterError} />
  484. </div>
  485. <div className="flex items-center justify-between">
  486. <span className="text-sm text-white">{t('notifications.lowFilament')}</span>
  487. <Toggle checked={onFilamentLow} onChange={setOnFilamentLow} />
  488. </div>
  489. <div className="flex items-center justify-between">
  490. <span className="text-sm text-white">{t('notifications.maintenance')}</span>
  491. <Toggle checked={onMaintenanceDue} onChange={setOnMaintenanceDue} />
  492. </div>
  493. </div>
  494. </div>
  495. </div>
  496. {/* Actions */}
  497. <div className="flex gap-3 pt-2">
  498. <Button
  499. type="button"
  500. variant="secondary"
  501. onClick={onClose}
  502. className="flex-1"
  503. >
  504. {t('notifications.cancel')}
  505. </Button>
  506. <Button
  507. type="submit"
  508. disabled={isPending}
  509. className="flex-1"
  510. >
  511. {isPending ? (
  512. <Loader2 className="w-4 h-4 animate-spin" />
  513. ) : (
  514. <Save className="w-4 h-4" />
  515. )}
  516. {isEditing ? t('notifications.save') : t('notifications.add')}
  517. </Button>
  518. </div>
  519. </form>
  520. </div>
  521. </div>
  522. );
  523. }