SettingsPage.tsx 205 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395
  1. import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
  2. import { Loader2, Plus, Plug, AlertTriangle, RotateCcw, Bell, Download, RefreshCw, ExternalLink, Globe, Droplets, Thermometer, FileText, Edit2, Send, CheckCircle, XCircle, History, Trash2, Zap, TrendingUp, Calendar, DollarSign, Power, PowerOff, Key, Copy, Database, X, Shield, Printer, Cylinder, Wifi, Home, Video, Users, Lock, Unlock, ChevronDown, ChevronRight, Check, Save } from 'lucide-react';
  3. import { useTranslation } from 'react-i18next';
  4. import { useNavigate, useSearchParams } from 'react-router-dom';
  5. import { api } from '../api/client';
  6. import { useAuth } from '../contexts/AuthContext';
  7. import { formatDateOnly } from '../utils/date';
  8. import type { AppSettings, AppSettingsUpdate, SmartPlug, SmartPlugStatus, NotificationProvider, NotificationTemplate, UpdateStatus, GitHubBackupStatus, CloudAuthStatus, UserCreate, UserUpdate, UserResponse, Group, GroupCreate, GroupUpdate, Permission, PermissionCategory } from '../api/client';
  9. import { Card, CardContent, CardHeader } from '../components/Card';
  10. import { Button } from '../components/Button';
  11. import { SmartPlugCard } from '../components/SmartPlugCard';
  12. import { AddSmartPlugModal } from '../components/AddSmartPlugModal';
  13. import { NotificationProviderCard } from '../components/NotificationProviderCard';
  14. import { AddNotificationModal } from '../components/AddNotificationModal';
  15. import { NotificationTemplateEditor } from '../components/NotificationTemplateEditor';
  16. import { NotificationLogViewer } from '../components/NotificationLogViewer';
  17. import { ConfirmModal } from '../components/ConfirmModal';
  18. import { SpoolmanSettings } from '../components/SpoolmanSettings';
  19. import { ExternalLinksSettings } from '../components/ExternalLinksSettings';
  20. import { VirtualPrinterSettings } from '../components/VirtualPrinterSettings';
  21. import { GitHubBackupSettings } from '../components/GitHubBackupSettings';
  22. import { APIBrowser } from '../components/APIBrowser';
  23. import { virtualPrinterApi } from '../api/client';
  24. import { defaultNavItems, getDefaultView, setDefaultView } from '../components/Layout';
  25. import { availableLanguages } from '../i18n';
  26. import { useToast } from '../contexts/ToastContext';
  27. import { useTheme, type ThemeStyle, type DarkBackground, type LightBackground, type ThemeAccent } from '../contexts/ThemeContext';
  28. import { useState, useEffect, useRef, useCallback } from 'react';
  29. import { Palette } from 'lucide-react';
  30. const validTabs = ['general', 'network', 'plugs', 'notifications', 'filament', 'apikeys', 'virtual-printer', 'users', 'backup'] as const;
  31. type TabType = typeof validTabs[number];
  32. export function SettingsPage() {
  33. const queryClient = useQueryClient();
  34. const navigate = useNavigate();
  35. const [searchParams, setSearchParams] = useSearchParams();
  36. const { t, i18n } = useTranslation();
  37. const { showToast } = useToast();
  38. const { authEnabled, user, refreshAuth } = useAuth();
  39. const {
  40. mode,
  41. darkStyle, darkBackground, darkAccent,
  42. lightStyle, lightBackground, lightAccent,
  43. setDarkStyle, setDarkBackground, setDarkAccent,
  44. setLightStyle, setLightBackground, setLightAccent,
  45. } = useTheme();
  46. const [localSettings, setLocalSettings] = useState<AppSettings | null>(null);
  47. const [showPlugModal, setShowPlugModal] = useState(false);
  48. const [editingPlug, setEditingPlug] = useState<SmartPlug | null>(null);
  49. const [showNotificationModal, setShowNotificationModal] = useState(false);
  50. const [editingProvider, setEditingProvider] = useState<NotificationProvider | null>(null);
  51. const [editingTemplate, setEditingTemplate] = useState<NotificationTemplate | null>(null);
  52. const [showLogViewer, setShowLogViewer] = useState(false);
  53. const [defaultView, setDefaultViewState] = useState<string>(getDefaultView());
  54. // Initialize tab from URL params
  55. const tabParam = searchParams.get('tab');
  56. const initialTab = tabParam && validTabs.includes(tabParam as TabType) ? tabParam as TabType : 'general';
  57. const [activeTab, setActiveTab] = useState<TabType>(initialTab);
  58. // Update URL when tab changes
  59. const handleTabChange = (tab: TabType) => {
  60. setActiveTab(tab);
  61. if (tab === 'general') {
  62. searchParams.delete('tab');
  63. } else {
  64. searchParams.set('tab', tab);
  65. }
  66. setSearchParams(searchParams, { replace: true });
  67. };
  68. const [showCreateAPIKey, setShowCreateAPIKey] = useState(false);
  69. const [newAPIKeyName, setNewAPIKeyName] = useState('');
  70. const [newAPIKeyPermissions, setNewAPIKeyPermissions] = useState({
  71. can_queue: true,
  72. can_control_printer: false,
  73. can_read_status: true,
  74. });
  75. const [createdAPIKey, setCreatedAPIKey] = useState<string | null>(null);
  76. const [showDeleteAPIKeyConfirm, setShowDeleteAPIKeyConfirm] = useState<number | null>(null);
  77. const [testApiKey, setTestApiKey] = useState('');
  78. // Confirm modal states
  79. const [showClearLogsConfirm, setShowClearLogsConfirm] = useState(false);
  80. const [showClearStorageConfirm, setShowClearStorageConfirm] = useState(false);
  81. const [showBulkPlugConfirm, setShowBulkPlugConfirm] = useState<'on' | 'off' | null>(null);
  82. const [showReleaseNotes, setShowReleaseNotes] = useState(false);
  83. const [showDisableAuthConfirm, setShowDisableAuthConfirm] = useState(false);
  84. const [showChangePasswordModal, setShowChangePasswordModal] = useState(false);
  85. const [changePasswordData, setChangePasswordData] = useState({ currentPassword: '', newPassword: '', confirmPassword: '' });
  86. const [changePasswordLoading, setChangePasswordLoading] = useState(false);
  87. // User management state
  88. const [showCreateUserModal, setShowCreateUserModal] = useState(false);
  89. const [showEditUserModal, setShowEditUserModal] = useState(false);
  90. const [editingUserId, setEditingUserId] = useState<number | null>(null);
  91. const [deleteUserId, setDeleteUserId] = useState<number | null>(null);
  92. const [deleteUserItemCounts, setDeleteUserItemCounts] = useState<{ archives: number; queue_items: number; library_files: number } | null>(null);
  93. const [deleteUserLoading, setDeleteUserLoading] = useState(false);
  94. const [userFormData, setUserFormData] = useState<{
  95. username: string;
  96. password: string;
  97. confirmPassword: string;
  98. role: string;
  99. group_ids: number[];
  100. }>({
  101. username: '',
  102. password: '',
  103. confirmPassword: '',
  104. role: 'user',
  105. group_ids: [],
  106. });
  107. // Group management state
  108. const [showCreateGroupModal, setShowCreateGroupModal] = useState(false);
  109. const [editingGroup, setEditingGroup] = useState<Group | null>(null);
  110. const [deleteGroupId, setDeleteGroupId] = useState<number | null>(null);
  111. const [groupFormData, setGroupFormData] = useState<{
  112. name: string;
  113. description: string;
  114. permissions: Permission[];
  115. }>({
  116. name: '',
  117. description: '',
  118. permissions: [],
  119. });
  120. const [expandedCategories, setExpandedCategories] = useState<Set<string>>(new Set());
  121. // Home Assistant test connection state
  122. const [haTestResult, setHaTestResult] = useState<{ success: boolean; message: string | null; error: string | null } | null>(null);
  123. const [haTestLoading, setHaTestLoading] = useState(false);
  124. // External camera test state
  125. const [extCameraTestResults, setExtCameraTestResults] = useState<Record<number, { success: boolean; error?: string; resolution?: string } | null>>({});
  126. const [extCameraTestLoading, setExtCameraTestLoading] = useState<Record<number, boolean>>({});
  127. const handleDefaultViewChange = (path: string) => {
  128. setDefaultViewState(path);
  129. setDefaultView(path);
  130. showToast(t('settings.toast.settingsSaved'), 'success');
  131. };
  132. const handleResetSidebarOrder = () => {
  133. localStorage.removeItem('sidebarOrder');
  134. window.location.reload();
  135. };
  136. const { data: settings, isLoading } = useQuery({
  137. queryKey: ['settings'],
  138. queryFn: api.getSettings,
  139. });
  140. const { data: smartPlugs, isLoading: plugsLoading } = useQuery({
  141. queryKey: ['smart-plugs'],
  142. queryFn: api.getSmartPlugs,
  143. });
  144. // Fetch energy data for all smart plugs when on the plugs tab
  145. const { data: plugEnergySummary, isLoading: energyLoading } = useQuery({
  146. queryKey: ['smart-plugs-energy', smartPlugs?.map(p => p.id)],
  147. queryFn: async () => {
  148. if (!smartPlugs || smartPlugs.length === 0) return null;
  149. const statuses = await Promise.all(
  150. smartPlugs.filter(p => p.enabled).map(async (plug) => {
  151. try {
  152. const status = await api.getSmartPlugStatus(plug.id);
  153. return { plug, status };
  154. } catch {
  155. return { plug, status: null as SmartPlugStatus | null };
  156. }
  157. })
  158. );
  159. // Aggregate energy data
  160. let totalPower = 0;
  161. let totalToday = 0;
  162. let totalYesterday = 0;
  163. let totalLifetime = 0;
  164. let reachableCount = 0;
  165. for (const { plug, status } of statuses) {
  166. // For MQTT plugs, consider reachable if we have power data
  167. const hasMqttData = plug.plug_type === 'mqtt' && (status?.energy?.power != null);
  168. const isReachable = (status?.reachable || hasMqttData) && status?.energy;
  169. if (isReachable) {
  170. reachableCount++;
  171. if (status.energy?.power != null) totalPower += status.energy.power;
  172. if (status.energy?.today != null) totalToday += status.energy.today;
  173. if (status.energy?.yesterday != null) totalYesterday += status.energy.yesterday;
  174. if (status.energy?.total != null) totalLifetime += status.energy.total;
  175. }
  176. }
  177. return {
  178. totalPower,
  179. totalToday,
  180. totalYesterday,
  181. totalLifetime,
  182. reachableCount,
  183. totalPlugs: smartPlugs.filter(p => p.enabled).length,
  184. };
  185. },
  186. enabled: activeTab === 'plugs' && !!smartPlugs && smartPlugs.length > 0,
  187. refetchInterval: activeTab === 'plugs' ? 10000 : false, // Refresh every 10s when on plugs tab
  188. });
  189. const { data: notificationProviders, isLoading: providersLoading } = useQuery({
  190. queryKey: ['notification-providers'],
  191. queryFn: api.getNotificationProviders,
  192. });
  193. const { data: apiKeys, isLoading: apiKeysLoading } = useQuery({
  194. queryKey: ['api-keys'],
  195. queryFn: api.getAPIKeys,
  196. });
  197. const createAPIKeyMutation = useMutation({
  198. mutationFn: (data: { name: string; can_queue: boolean; can_control_printer: boolean; can_read_status: boolean }) =>
  199. api.createAPIKey(data),
  200. onSuccess: (data) => {
  201. setCreatedAPIKey(data.key || null);
  202. setShowCreateAPIKey(false);
  203. setNewAPIKeyName('');
  204. queryClient.invalidateQueries({ queryKey: ['api-keys'] });
  205. showToast(t('settings.toast.apiKeyCreated'));
  206. },
  207. onError: (error: Error) => {
  208. showToast(`Failed to create API key: ${error.message}`, 'error');
  209. },
  210. });
  211. const deleteAPIKeyMutation = useMutation({
  212. mutationFn: (id: number) => api.deleteAPIKey(id),
  213. onSuccess: () => {
  214. queryClient.invalidateQueries({ queryKey: ['api-keys'] });
  215. showToast(t('settings.toast.apiKeyDeleted'));
  216. },
  217. onError: (error: Error) => {
  218. showToast(`Failed to delete API key: ${error.message}`, 'error');
  219. },
  220. });
  221. const { data: printers } = useQuery({
  222. queryKey: ['printers'],
  223. queryFn: api.getPrinters,
  224. });
  225. const { data: notificationTemplates, isLoading: templatesLoading } = useQuery({
  226. queryKey: ['notification-templates'],
  227. queryFn: api.getNotificationTemplates,
  228. });
  229. // Virtual printer status for tab indicator
  230. const { data: virtualPrinterSettings } = useQuery({
  231. queryKey: ['virtual-printer-settings'],
  232. queryFn: virtualPrinterApi.getSettings,
  233. refetchInterval: 10000,
  234. });
  235. const virtualPrinterRunning = virtualPrinterSettings?.status?.running ?? false;
  236. const { data: ffmpegStatus } = useQuery({
  237. queryKey: ['ffmpeg-status'],
  238. queryFn: api.checkFfmpeg,
  239. });
  240. const { data: versionInfo } = useQuery({
  241. queryKey: ['version'],
  242. queryFn: api.getVersion,
  243. });
  244. const { data: updateCheck, refetch: refetchUpdateCheck, isRefetching: isCheckingUpdate } = useQuery({
  245. queryKey: ['updateCheck'],
  246. queryFn: api.checkForUpdates,
  247. staleTime: 5 * 60 * 1000,
  248. });
  249. const { data: updateStatus, refetch: refetchUpdateStatus } = useQuery({
  250. queryKey: ['updateStatus'],
  251. queryFn: api.getUpdateStatus,
  252. refetchInterval: (query) => {
  253. const status = query.state.data as UpdateStatus | undefined;
  254. // Poll while update is in progress
  255. if (status?.status === 'downloading' || status?.status === 'installing') {
  256. return 1000;
  257. }
  258. return false;
  259. },
  260. });
  261. // MQTT status for Network tab
  262. const { data: mqttStatus } = useQuery({
  263. queryKey: ['mqtt-status'],
  264. queryFn: api.getMQTTStatus,
  265. refetchInterval: activeTab === 'network' ? 5000 : false, // Poll every 5s when on Network tab
  266. });
  267. // GitHub backup status for Backup tab indicator
  268. const { data: githubBackupStatus } = useQuery<GitHubBackupStatus>({
  269. queryKey: ['github-backup-status'],
  270. queryFn: api.getGitHubBackupStatus,
  271. });
  272. // Cloud auth status for Backup tab indicator
  273. const { data: cloudAuthStatus } = useQuery<CloudAuthStatus>({
  274. queryKey: ['cloud-status'],
  275. queryFn: api.getCloudStatus,
  276. });
  277. // User management queries and mutations
  278. const { hasPermission } = useAuth();
  279. const { data: usersData = [], isLoading: usersLoading } = useQuery({
  280. queryKey: ['users'],
  281. queryFn: () => api.getUsers(),
  282. enabled: authEnabled && hasPermission('users:read'),
  283. });
  284. const { data: groupsData = [], isLoading: groupsLoading } = useQuery({
  285. queryKey: ['groups'],
  286. queryFn: () => api.getGroups(),
  287. enabled: authEnabled && hasPermission('groups:read'),
  288. });
  289. const { data: permissionsData } = useQuery({
  290. queryKey: ['permissions'],
  291. queryFn: () => api.getPermissions(),
  292. enabled: authEnabled && hasPermission('groups:read'),
  293. });
  294. const createUserMutation = useMutation({
  295. mutationFn: (data: UserCreate) => api.createUser(data),
  296. onSuccess: () => {
  297. queryClient.invalidateQueries({ queryKey: ['users'] });
  298. queryClient.invalidateQueries({ queryKey: ['groups'] });
  299. setShowCreateUserModal(false);
  300. setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
  301. showToast(t('settings.toast.userCreated'));
  302. },
  303. onError: (error: Error) => {
  304. showToast(error.message, 'error');
  305. },
  306. });
  307. const updateUserMutation = useMutation({
  308. mutationFn: ({ id, data }: { id: number; data: UserUpdate }) => api.updateUser(id, data),
  309. onSuccess: () => {
  310. queryClient.invalidateQueries({ queryKey: ['users'] });
  311. queryClient.invalidateQueries({ queryKey: ['groups'] });
  312. setShowEditUserModal(false);
  313. setEditingUserId(null);
  314. setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
  315. showToast(t('settings.toast.userUpdated'));
  316. },
  317. onError: (error: Error) => {
  318. showToast(error.message, 'error');
  319. },
  320. });
  321. const deleteUserMutation = useMutation({
  322. mutationFn: ({ id, deleteItems }: { id: number; deleteItems: boolean }) => api.deleteUser(id, deleteItems),
  323. onSuccess: () => {
  324. queryClient.invalidateQueries({ queryKey: ['users'] });
  325. showToast(t('settings.toast.userDeleted'));
  326. setDeleteUserId(null);
  327. setDeleteUserItemCounts(null);
  328. },
  329. onError: (error: Error) => {
  330. showToast(error.message, 'error');
  331. },
  332. });
  333. // Function to initiate user deletion with item count check
  334. const handleDeleteUserClick = async (userId: number) => {
  335. setDeleteUserId(userId);
  336. setDeleteUserLoading(true);
  337. try {
  338. const counts = await api.getUserItemsCount(userId);
  339. setDeleteUserItemCounts(counts);
  340. } catch {
  341. // If we can't get counts, just proceed without showing item options
  342. setDeleteUserItemCounts({ archives: 0, queue_items: 0, library_files: 0 });
  343. } finally {
  344. setDeleteUserLoading(false);
  345. }
  346. };
  347. const createGroupMutation = useMutation({
  348. mutationFn: (data: GroupCreate) => api.createGroup(data),
  349. onSuccess: () => {
  350. queryClient.invalidateQueries({ queryKey: ['groups'] });
  351. setShowCreateGroupModal(false);
  352. resetGroupForm();
  353. showToast(t('settings.toast.groupCreated'));
  354. },
  355. onError: (error: Error) => {
  356. showToast(error.message, 'error');
  357. },
  358. });
  359. const updateGroupMutation = useMutation({
  360. mutationFn: ({ id, data }: { id: number; data: GroupUpdate }) => api.updateGroup(id, data),
  361. onSuccess: () => {
  362. queryClient.invalidateQueries({ queryKey: ['groups'] });
  363. setEditingGroup(null);
  364. resetGroupForm();
  365. showToast(t('settings.toast.groupUpdated'));
  366. },
  367. onError: (error: Error) => {
  368. showToast(error.message, 'error');
  369. },
  370. });
  371. const deleteGroupMutation = useMutation({
  372. mutationFn: (id: number) => api.deleteGroup(id),
  373. onSuccess: () => {
  374. queryClient.invalidateQueries({ queryKey: ['groups'] });
  375. showToast(t('settings.toast.groupDeleted'));
  376. },
  377. onError: (error: Error) => {
  378. showToast(error.message, 'error');
  379. },
  380. });
  381. // User management handlers
  382. const handleCreateUser = () => {
  383. if (!userFormData.username || !userFormData.password) {
  384. showToast(t('settings.toast.fillRequiredFields'), 'error');
  385. return;
  386. }
  387. if (userFormData.password !== userFormData.confirmPassword) {
  388. showToast(t('settings.toast.passwordsDoNotMatch'), 'error');
  389. return;
  390. }
  391. if (userFormData.password.length < 6) {
  392. showToast(t('settings.toast.passwordTooShort'), 'error');
  393. return;
  394. }
  395. createUserMutation.mutate({
  396. username: userFormData.username,
  397. password: userFormData.password,
  398. role: userFormData.role,
  399. group_ids: userFormData.group_ids.length > 0 ? userFormData.group_ids : undefined,
  400. });
  401. };
  402. const handleUpdateUser = (id: number) => {
  403. if (userFormData.password) {
  404. if (userFormData.password !== userFormData.confirmPassword) {
  405. showToast(t('settings.toast.passwordsDoNotMatch'), 'error');
  406. return;
  407. }
  408. if (userFormData.password.length < 6) {
  409. showToast(t('settings.toast.passwordTooShort'), 'error');
  410. return;
  411. }
  412. }
  413. const updateData: UserUpdate = {
  414. username: userFormData.username || undefined,
  415. password: userFormData.password || undefined,
  416. role: userFormData.role,
  417. group_ids: userFormData.group_ids,
  418. };
  419. if (!updateData.password) {
  420. delete updateData.password;
  421. }
  422. updateUserMutation.mutate({ id, data: updateData });
  423. };
  424. const startEditUser = (userToEdit: UserResponse) => {
  425. setEditingUserId(userToEdit.id);
  426. setUserFormData({
  427. username: userToEdit.username,
  428. password: '',
  429. confirmPassword: '',
  430. role: userToEdit.role,
  431. group_ids: userToEdit.groups?.map(g => g.id) || [],
  432. });
  433. setShowEditUserModal(true);
  434. };
  435. const toggleUserGroup = (groupId: number) => {
  436. setUserFormData(prev => ({
  437. ...prev,
  438. group_ids: prev.group_ids.includes(groupId)
  439. ? prev.group_ids.filter(id => id !== groupId)
  440. : [...prev.group_ids, groupId],
  441. }));
  442. };
  443. // Group management handlers
  444. const resetGroupForm = () => {
  445. setGroupFormData({ name: '', description: '', permissions: [] });
  446. setExpandedCategories(new Set());
  447. };
  448. const handleCreateGroup = () => {
  449. if (!groupFormData.name.trim()) {
  450. showToast(t('settings.toast.enterGroupName'), 'error');
  451. return;
  452. }
  453. createGroupMutation.mutate({
  454. name: groupFormData.name,
  455. description: groupFormData.description || undefined,
  456. permissions: groupFormData.permissions,
  457. });
  458. };
  459. const handleUpdateGroup = () => {
  460. if (!editingGroup) return;
  461. if (!groupFormData.name.trim()) {
  462. showToast(t('settings.toast.enterGroupName'), 'error');
  463. return;
  464. }
  465. updateGroupMutation.mutate({
  466. id: editingGroup.id,
  467. data: {
  468. name: groupFormData.name !== editingGroup.name ? groupFormData.name : undefined,
  469. description: groupFormData.description,
  470. permissions: groupFormData.permissions,
  471. },
  472. });
  473. };
  474. const startEditGroup = (group: Group) => {
  475. setEditingGroup(group);
  476. setGroupFormData({
  477. name: group.name,
  478. description: group.description || '',
  479. permissions: group.permissions,
  480. });
  481. const cats = new Set<string>();
  482. permissionsData?.categories.forEach((cat) => {
  483. if (cat.permissions.some((p) => group.permissions.includes(p.value))) {
  484. cats.add(cat.name);
  485. }
  486. });
  487. setExpandedCategories(cats);
  488. };
  489. const toggleCategory = (categoryName: string) => {
  490. setExpandedCategories((prev) => {
  491. const next = new Set(prev);
  492. if (next.has(categoryName)) {
  493. next.delete(categoryName);
  494. } else {
  495. next.add(categoryName);
  496. }
  497. return next;
  498. });
  499. };
  500. const togglePermission = (permission: Permission) => {
  501. setGroupFormData((prev) => {
  502. const permissions = prev.permissions.includes(permission)
  503. ? prev.permissions.filter((p) => p !== permission)
  504. : [...prev.permissions, permission];
  505. return { ...prev, permissions };
  506. });
  507. };
  508. const toggleCategoryPermissions = (category: PermissionCategory, checked: boolean) => {
  509. setGroupFormData((prev) => {
  510. const categoryPerms = category.permissions.map((p) => p.value);
  511. const otherPerms = prev.permissions.filter((p) => !categoryPerms.includes(p));
  512. const permissions = checked ? [...otherPerms, ...categoryPerms] : otherPerms;
  513. return { ...prev, permissions };
  514. });
  515. };
  516. const isCategoryFullySelected = (category: PermissionCategory) => {
  517. return category.permissions.every((p) => groupFormData.permissions.includes(p.value));
  518. };
  519. const isCategoryPartiallySelected = (category: PermissionCategory) => {
  520. const selected = category.permissions.filter((p) => groupFormData.permissions.includes(p.value));
  521. return selected.length > 0 && selected.length < category.permissions.length;
  522. };
  523. const applyUpdateMutation = useMutation({
  524. mutationFn: api.applyUpdate,
  525. onSuccess: (data) => {
  526. if (data.is_docker) {
  527. showToast(data.message, 'error');
  528. } else {
  529. refetchUpdateStatus();
  530. }
  531. },
  532. });
  533. // Test all notification providers
  534. const [testAllResult, setTestAllResult] = useState<{
  535. tested: number;
  536. success: number;
  537. failed: number;
  538. results: Array<{
  539. provider_id: number;
  540. provider_name: string;
  541. provider_type: string;
  542. success: boolean;
  543. message: string;
  544. }>;
  545. } | null>(null);
  546. const testAllMutation = useMutation({
  547. mutationFn: api.testAllNotificationProviders,
  548. onSuccess: (data) => {
  549. setTestAllResult(data);
  550. queryClient.invalidateQueries({ queryKey: ['notification-providers'] });
  551. if (data.failed === 0) {
  552. showToast(`All ${data.tested} providers tested successfully!`, 'success');
  553. } else {
  554. showToast(`${data.success}/${data.tested} providers succeeded`, data.failed > 0 ? 'error' : 'success');
  555. }
  556. },
  557. onError: (error: Error) => {
  558. showToast(`Failed to test providers: ${error.message}`, 'error');
  559. },
  560. });
  561. // Bulk action for smart plugs
  562. const bulkPlugActionMutation = useMutation({
  563. mutationFn: async (action: 'on' | 'off') => {
  564. if (!smartPlugs) return { success: 0, failed: 0 };
  565. const enabledPlugs = smartPlugs.filter(p => p.enabled);
  566. const results = await Promise.all(
  567. enabledPlugs.map(async (plug) => {
  568. try {
  569. await api.controlSmartPlug(plug.id, action);
  570. return { success: true };
  571. } catch {
  572. return { success: false };
  573. }
  574. })
  575. );
  576. return {
  577. success: results.filter(r => r.success).length,
  578. failed: results.filter(r => !r.success).length,
  579. };
  580. },
  581. onSuccess: (data, action) => {
  582. queryClient.invalidateQueries({ queryKey: ['smart-plugs'] });
  583. queryClient.invalidateQueries({ queryKey: ['smart-plugs-energy'] });
  584. if (data.failed === 0) {
  585. showToast(`All ${data.success} plugs turned ${action}`, 'success');
  586. } else {
  587. showToast(`${data.success} plugs turned ${action}, ${data.failed} failed`, 'error');
  588. }
  589. },
  590. onError: (error: Error) => {
  591. showToast(`Failed: ${error.message}`, 'error');
  592. },
  593. });
  594. // Ref for debounce timeout
  595. const saveTimeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null);
  596. const isSavingRef = useRef(false);
  597. const isInitialLoadRef = useRef(true);
  598. // Sync local state when settings load
  599. useEffect(() => {
  600. if (settings && !localSettings) {
  601. // Auto-detect external_url from browser if not set
  602. const settingsWithExternalUrl = {
  603. ...settings,
  604. external_url: settings.external_url || window.location.origin,
  605. };
  606. setLocalSettings(settingsWithExternalUrl);
  607. // Mark initial load complete after a short delay
  608. setTimeout(() => {
  609. isInitialLoadRef.current = false;
  610. }, 100);
  611. }
  612. }, [settings, localSettings]);
  613. const updateMutation = useMutation({
  614. mutationFn: api.updateSettings,
  615. onSuccess: (data) => {
  616. queryClient.setQueryData(['settings'], data);
  617. // Sync localSettings with the saved data to prevent re-triggering saves
  618. setLocalSettings(data);
  619. // Invalidate archive stats to reflect energy tracking mode change
  620. queryClient.invalidateQueries({ queryKey: ['archiveStats'] });
  621. showToast(t('settings.toast.settingsSaved'), 'success');
  622. },
  623. onError: (error: Error) => {
  624. showToast(`Failed to save: ${error.message}`, 'error');
  625. },
  626. onSettled: () => {
  627. // Reset saving flag when mutation completes (success or error)
  628. isSavingRef.current = false;
  629. },
  630. });
  631. const updatePrinterMutation = useMutation({
  632. mutationFn: ({ id, data }: { id: number; data: Partial<{ external_camera_url: string | null; external_camera_type: string | null; external_camera_enabled: boolean }> }) =>
  633. api.updatePrinter(id, data),
  634. onSuccess: () => {
  635. queryClient.invalidateQueries({ queryKey: ['printers'] });
  636. showToast(t('settings.toast.cameraSettingsSaved'), 'success');
  637. },
  638. onError: (error: Error) => {
  639. showToast(`Failed to update printer: ${error.message}`, 'error');
  640. },
  641. });
  642. // Debounced auto-save when localSettings change
  643. useEffect(() => {
  644. // Skip if initial load or no settings
  645. if (isInitialLoadRef.current || !localSettings || !settings) {
  646. return;
  647. }
  648. // Check if there are actual changes
  649. const hasChanges =
  650. settings.auto_archive !== localSettings.auto_archive ||
  651. settings.save_thumbnails !== localSettings.save_thumbnails ||
  652. settings.capture_finish_photo !== localSettings.capture_finish_photo ||
  653. settings.default_filament_cost !== localSettings.default_filament_cost ||
  654. settings.currency !== localSettings.currency ||
  655. settings.energy_cost_per_kwh !== localSettings.energy_cost_per_kwh ||
  656. settings.energy_tracking_mode !== localSettings.energy_tracking_mode ||
  657. settings.check_updates !== localSettings.check_updates ||
  658. (settings.check_printer_firmware ?? true) !== (localSettings.check_printer_firmware ?? true) ||
  659. settings.notification_language !== localSettings.notification_language ||
  660. settings.ams_humidity_good !== localSettings.ams_humidity_good ||
  661. settings.ams_humidity_fair !== localSettings.ams_humidity_fair ||
  662. settings.ams_temp_good !== localSettings.ams_temp_good ||
  663. settings.ams_temp_fair !== localSettings.ams_temp_fair ||
  664. settings.ams_history_retention_days !== localSettings.ams_history_retention_days ||
  665. settings.per_printer_mapping_expanded !== localSettings.per_printer_mapping_expanded ||
  666. settings.date_format !== localSettings.date_format ||
  667. settings.time_format !== localSettings.time_format ||
  668. settings.default_printer_id !== localSettings.default_printer_id ||
  669. settings.ftp_retry_enabled !== localSettings.ftp_retry_enabled ||
  670. settings.ftp_retry_count !== localSettings.ftp_retry_count ||
  671. settings.ftp_retry_delay !== localSettings.ftp_retry_delay ||
  672. settings.ftp_timeout !== localSettings.ftp_timeout ||
  673. settings.mqtt_enabled !== localSettings.mqtt_enabled ||
  674. settings.mqtt_broker !== localSettings.mqtt_broker ||
  675. settings.mqtt_port !== localSettings.mqtt_port ||
  676. settings.mqtt_username !== localSettings.mqtt_username ||
  677. settings.mqtt_password !== localSettings.mqtt_password ||
  678. settings.mqtt_topic_prefix !== localSettings.mqtt_topic_prefix ||
  679. settings.mqtt_use_tls !== localSettings.mqtt_use_tls ||
  680. settings.external_url !== localSettings.external_url ||
  681. settings.ha_enabled !== localSettings.ha_enabled ||
  682. settings.ha_url !== localSettings.ha_url ||
  683. settings.ha_token !== localSettings.ha_token ||
  684. (settings.library_archive_mode ?? 'ask') !== (localSettings.library_archive_mode ?? 'ask') ||
  685. Number(settings.library_disk_warning_gb ?? 5) !== Number(localSettings.library_disk_warning_gb ?? 5) ||
  686. (settings.camera_view_mode ?? 'window') !== (localSettings.camera_view_mode ?? 'window') ||
  687. settings.prometheus_enabled !== localSettings.prometheus_enabled ||
  688. settings.prometheus_token !== localSettings.prometheus_token;
  689. if (!hasChanges) {
  690. return;
  691. }
  692. // Don't queue more saves while one is in progress
  693. if (isSavingRef.current) {
  694. return;
  695. }
  696. // Clear existing timeout
  697. if (saveTimeoutRef.current) {
  698. clearTimeout(saveTimeoutRef.current);
  699. }
  700. // Set new debounced save (500ms delay)
  701. saveTimeoutRef.current = setTimeout(() => {
  702. // Skip if a save is already in progress
  703. if (isSavingRef.current) {
  704. return;
  705. }
  706. isSavingRef.current = true;
  707. // Only send the fields we manage on this page (exclude virtual_printer_* which are managed separately)
  708. const settingsToSave: AppSettingsUpdate = {
  709. auto_archive: localSettings.auto_archive,
  710. save_thumbnails: localSettings.save_thumbnails,
  711. capture_finish_photo: localSettings.capture_finish_photo,
  712. default_filament_cost: localSettings.default_filament_cost,
  713. currency: localSettings.currency,
  714. energy_cost_per_kwh: localSettings.energy_cost_per_kwh,
  715. energy_tracking_mode: localSettings.energy_tracking_mode,
  716. check_updates: localSettings.check_updates,
  717. check_printer_firmware: localSettings.check_printer_firmware,
  718. notification_language: localSettings.notification_language,
  719. ams_humidity_good: localSettings.ams_humidity_good,
  720. ams_humidity_fair: localSettings.ams_humidity_fair,
  721. ams_temp_good: localSettings.ams_temp_good,
  722. ams_temp_fair: localSettings.ams_temp_fair,
  723. ams_history_retention_days: localSettings.ams_history_retention_days,
  724. per_printer_mapping_expanded: localSettings.per_printer_mapping_expanded,
  725. date_format: localSettings.date_format,
  726. time_format: localSettings.time_format,
  727. default_printer_id: localSettings.default_printer_id,
  728. ftp_retry_enabled: localSettings.ftp_retry_enabled,
  729. ftp_retry_count: localSettings.ftp_retry_count,
  730. ftp_retry_delay: localSettings.ftp_retry_delay,
  731. ftp_timeout: localSettings.ftp_timeout,
  732. mqtt_enabled: localSettings.mqtt_enabled,
  733. mqtt_broker: localSettings.mqtt_broker,
  734. mqtt_port: localSettings.mqtt_port,
  735. mqtt_username: localSettings.mqtt_username,
  736. mqtt_password: localSettings.mqtt_password,
  737. mqtt_topic_prefix: localSettings.mqtt_topic_prefix,
  738. mqtt_use_tls: localSettings.mqtt_use_tls,
  739. external_url: localSettings.external_url,
  740. ha_enabled: localSettings.ha_enabled,
  741. ha_url: localSettings.ha_url,
  742. ha_token: localSettings.ha_token,
  743. library_archive_mode: localSettings.library_archive_mode,
  744. library_disk_warning_gb: localSettings.library_disk_warning_gb,
  745. camera_view_mode: localSettings.camera_view_mode,
  746. prometheus_enabled: localSettings.prometheus_enabled,
  747. prometheus_token: localSettings.prometheus_token,
  748. };
  749. updateMutation.mutate(settingsToSave);
  750. }, 500);
  751. // Cleanup on unmount or when localSettings changes again
  752. return () => {
  753. if (saveTimeoutRef.current) {
  754. clearTimeout(saveTimeoutRef.current);
  755. }
  756. };
  757. }, [localSettings, settings, updateMutation]);
  758. const updateSetting = useCallback(<K extends keyof AppSettings>(key: K, value: AppSettings[K]) => {
  759. setLocalSettings(prev => prev ? { ...prev, [key]: value } : null);
  760. }, []);
  761. const handleTestExternalCamera = async (printerId: number, url: string, cameraType: string) => {
  762. if (!url) {
  763. showToast(t('settings.toast.enterCameraUrl'), 'error');
  764. return;
  765. }
  766. setExtCameraTestLoading(prev => ({ ...prev, [printerId]: true }));
  767. setExtCameraTestResults(prev => ({ ...prev, [printerId]: null }));
  768. try {
  769. const result = await api.testExternalCamera(printerId, url, cameraType);
  770. setExtCameraTestResults(prev => ({ ...prev, [printerId]: result }));
  771. if (result.success) {
  772. showToast(t('settings.toast.cameraConnected', { resolution: result.resolution || '' }), 'success');
  773. } else {
  774. showToast(result.error || t('settings.toast.connectionFailed'), 'error');
  775. }
  776. } catch (error) {
  777. const message = error instanceof Error ? error.message : t('settings.toast.testFailed');
  778. setExtCameraTestResults(prev => ({ ...prev, [printerId]: { success: false, error: message } }));
  779. showToast(message, 'error');
  780. } finally {
  781. setExtCameraTestLoading(prev => ({ ...prev, [printerId]: false }));
  782. }
  783. };
  784. // Local state for camera URL inputs (to avoid saving on every keystroke)
  785. const [localCameraUrls, setLocalCameraUrls] = useState<Record<number, string>>({});
  786. const cameraUrlSaveTimeoutRef = useRef<Record<number, ReturnType<typeof setTimeout>>>({});
  787. const initializedPrinterUrlsRef = useRef<Set<number>>(new Set());
  788. // Initialize local camera URLs from printer data
  789. useEffect(() => {
  790. if (printers) {
  791. const urls: Record<number, string> = {};
  792. printers.forEach(p => {
  793. if (p.external_camera_url && !initializedPrinterUrlsRef.current.has(p.id)) {
  794. urls[p.id] = p.external_camera_url;
  795. initializedPrinterUrlsRef.current.add(p.id);
  796. }
  797. });
  798. if (Object.keys(urls).length > 0) {
  799. setLocalCameraUrls(prev => ({ ...prev, ...urls }));
  800. }
  801. }
  802. }, [printers]);
  803. const handleCameraUrlChange = (printerId: number, url: string) => {
  804. // Update local state immediately for responsive UI
  805. setLocalCameraUrls(prev => ({ ...prev, [printerId]: url }));
  806. // Clear existing timeout for this printer
  807. if (cameraUrlSaveTimeoutRef.current[printerId]) {
  808. clearTimeout(cameraUrlSaveTimeoutRef.current[printerId]);
  809. }
  810. // Debounce the save (800ms delay)
  811. cameraUrlSaveTimeoutRef.current[printerId] = setTimeout(() => {
  812. updatePrinterMutation.mutate({
  813. id: printerId,
  814. data: { external_camera_url: url || null }
  815. });
  816. }, 800);
  817. };
  818. const handleUpdatePrinterCamera = (printerId: number, updates: { type?: string; enabled?: boolean }) => {
  819. const data: Partial<{ external_camera_type: string | null; external_camera_enabled: boolean }> = {};
  820. if (updates.type !== undefined) data.external_camera_type = updates.type || null;
  821. if (updates.enabled !== undefined) data.external_camera_enabled = updates.enabled;
  822. updatePrinterMutation.mutate({ id: printerId, data });
  823. };
  824. if (isLoading || !localSettings) {
  825. return (
  826. <div className="p-4 md:p-8 flex justify-center">
  827. <Loader2 className="w-8 h-8 text-bambu-green animate-spin" />
  828. </div>
  829. );
  830. }
  831. return (
  832. <div className="p-4 md:p-8">
  833. <div className="mb-8">
  834. <h1 className="text-2xl font-bold text-white">{t('settings.title')}</h1>
  835. <p className="text-bambu-gray">{t('settings.configureBambuddy')}</p>
  836. </div>
  837. {/* Tab Navigation */}
  838. <div className="flex gap-1 mb-6 border-b border-bambu-dark-tertiary overflow-x-auto">
  839. <button
  840. onClick={() => handleTabChange('general')}
  841. className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px ${
  842. activeTab === 'general'
  843. ? 'text-bambu-green border-bambu-green'
  844. : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
  845. }`}
  846. >
  847. {t('settings.tabs.general')}
  848. </button>
  849. <button
  850. onClick={() => handleTabChange('plugs')}
  851. className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px flex items-center gap-2 ${
  852. activeTab === 'plugs'
  853. ? 'text-bambu-green border-bambu-green'
  854. : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
  855. }`}
  856. >
  857. <Plug className="w-4 h-4" />
  858. {t('settings.tabs.smartPlugs')}
  859. {smartPlugs && smartPlugs.length > 0 && (
  860. <span className="text-xs bg-bambu-dark-tertiary px-1.5 py-0.5 rounded-full">
  861. {smartPlugs.length}
  862. </span>
  863. )}
  864. </button>
  865. <button
  866. onClick={() => handleTabChange('notifications')}
  867. className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px flex items-center gap-2 ${
  868. activeTab === 'notifications'
  869. ? 'text-bambu-green border-bambu-green'
  870. : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
  871. }`}
  872. >
  873. <Bell className="w-4 h-4" />
  874. {t('settings.tabs.notifications')}
  875. {notificationProviders && notificationProviders.length > 0 && (
  876. <span className="text-xs bg-bambu-dark-tertiary px-1.5 py-0.5 rounded-full">
  877. {notificationProviders.length}
  878. </span>
  879. )}
  880. </button>
  881. <button
  882. onClick={() => handleTabChange('filament')}
  883. className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px flex items-center gap-2 ${
  884. activeTab === 'filament'
  885. ? 'text-bambu-green border-bambu-green'
  886. : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
  887. }`}
  888. >
  889. <Cylinder className="w-4 h-4" />
  890. {t('settings.tabs.filament')}
  891. </button>
  892. <button
  893. onClick={() => handleTabChange('network')}
  894. className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px flex items-center gap-2 ${
  895. activeTab === 'network'
  896. ? 'text-bambu-green border-bambu-green'
  897. : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
  898. }`}
  899. >
  900. <Wifi className="w-4 h-4" />
  901. {t('settings.tabs.network')}
  902. <span className={`w-2 h-2 rounded-full ${mqttStatus?.enabled ? 'bg-green-400' : 'bg-gray-500'}`} />
  903. </button>
  904. <button
  905. onClick={() => handleTabChange('apikeys')}
  906. className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px flex items-center gap-2 ${
  907. activeTab === 'apikeys'
  908. ? 'text-bambu-green border-bambu-green'
  909. : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
  910. }`}
  911. >
  912. <Key className="w-4 h-4" />
  913. {t('settings.tabs.apiKeys')}
  914. {apiKeys && apiKeys.length > 0 && (
  915. <span className="text-xs bg-bambu-dark-tertiary px-1.5 py-0.5 rounded-full">
  916. {apiKeys.length}
  917. </span>
  918. )}
  919. </button>
  920. <button
  921. onClick={() => handleTabChange('virtual-printer')}
  922. className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px flex items-center gap-2 ${
  923. activeTab === 'virtual-printer'
  924. ? 'text-bambu-green border-bambu-green'
  925. : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
  926. }`}
  927. >
  928. <Printer className="w-4 h-4" />
  929. {t('settings.tabs.virtualPrinter')}
  930. <span className={`w-2 h-2 rounded-full ${virtualPrinterRunning ? 'bg-green-400' : 'bg-gray-500'}`} />
  931. </button>
  932. <button
  933. onClick={() => handleTabChange('users')}
  934. className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px flex items-center gap-2 ${
  935. activeTab === 'users'
  936. ? 'text-bambu-green border-bambu-green'
  937. : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
  938. }`}
  939. >
  940. <Users className="w-4 h-4" />
  941. {t('settings.tabs.users')}
  942. {authEnabled && (
  943. <span className="w-2 h-2 rounded-full bg-green-400" />
  944. )}
  945. </button>
  946. <button
  947. onClick={() => handleTabChange('backup')}
  948. className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px flex items-center gap-2 ${
  949. activeTab === 'backup'
  950. ? 'text-bambu-green border-bambu-green'
  951. : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
  952. }`}
  953. >
  954. <Database className="w-4 h-4" />
  955. {t('settings.tabs.backup')}
  956. <span className={`w-2 h-2 rounded-full ${cloudAuthStatus?.is_authenticated && githubBackupStatus?.configured && githubBackupStatus?.enabled ? 'bg-green-400' : 'bg-gray-500'}`} />
  957. </button>
  958. </div>
  959. {/* General Tab */}
  960. {activeTab === 'general' && (
  961. <div className="flex flex-col lg:flex-row gap-6 lg:gap-8">
  962. {/* Left Column - General Settings */}
  963. <div className="space-y-6 flex-1 lg:max-w-xl">
  964. <Card>
  965. <CardHeader>
  966. <h2 className="text-lg font-semibold text-white">{t('settings.general')}</h2>
  967. </CardHeader>
  968. <CardContent className="space-y-4">
  969. <div>
  970. <label className="block text-sm text-bambu-gray mb-1">
  971. <Globe className="w-4 h-4 inline mr-1" />
  972. {t('settings.language')}
  973. </label>
  974. <div className="relative">
  975. <select
  976. value={i18n.language}
  977. onChange={(e) => { i18n.changeLanguage(e.target.value); showToast(t('settings.toast.settingsSaved'), 'success'); }}
  978. className="w-full px-3 py-2 pr-10 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none appearance-none cursor-pointer"
  979. >
  980. {availableLanguages.map((lang) => (
  981. <option key={lang.code} value={lang.code}>
  982. {lang.nativeName} ({lang.name})
  983. </option>
  984. ))}
  985. </select>
  986. <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
  987. </div>
  988. <p className="text-xs text-bambu-gray mt-1">
  989. {t('settings.languageDescription')}
  990. </p>
  991. </div>
  992. <div>
  993. <label className="block text-sm text-bambu-gray mb-1">
  994. {t('settings.defaultView')}
  995. </label>
  996. <div className="relative">
  997. <select
  998. value={defaultView}
  999. onChange={(e) => handleDefaultViewChange(e.target.value)}
  1000. className="w-full px-3 py-2 pr-10 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none appearance-none cursor-pointer"
  1001. >
  1002. {defaultNavItems.map((item) => (
  1003. <option key={item.id} value={item.to}>
  1004. {t(item.labelKey)}
  1005. </option>
  1006. ))}
  1007. </select>
  1008. <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
  1009. </div>
  1010. <p className="text-xs text-bambu-gray mt-1">
  1011. {t('settings.defaultViewDescription')}
  1012. </p>
  1013. </div>
  1014. <div className="grid grid-cols-2 gap-3">
  1015. <div>
  1016. <label className="block text-sm text-bambu-gray mb-1">
  1017. Date Format
  1018. </label>
  1019. <div className="relative">
  1020. <select
  1021. value={localSettings.date_format || 'system'}
  1022. onChange={(e) => updateSetting('date_format', e.target.value as 'system' | 'us' | 'eu' | 'iso')}
  1023. className="w-full px-3 py-2 pr-10 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none appearance-none cursor-pointer"
  1024. >
  1025. <option value="system">{t('settings.systemDefault')}</option>
  1026. <option value="us">US (MM/DD/YYYY)</option>
  1027. <option value="eu">EU (DD/MM/YYYY)</option>
  1028. <option value="iso">ISO (YYYY-MM-DD)</option>
  1029. </select>
  1030. <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
  1031. </div>
  1032. </div>
  1033. <div>
  1034. <label className="block text-sm text-bambu-gray mb-1">
  1035. Time Format
  1036. </label>
  1037. <div className="relative">
  1038. <select
  1039. value={localSettings.time_format || 'system'}
  1040. onChange={(e) => updateSetting('time_format', e.target.value as 'system' | '12h' | '24h')}
  1041. className="w-full px-3 py-2 pr-10 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none appearance-none cursor-pointer"
  1042. >
  1043. <option value="system">{t('settings.systemDefault')}</option>
  1044. <option value="12h">12-hour (3:30 PM)</option>
  1045. <option value="24h">24-hour (15:30)</option>
  1046. </select>
  1047. <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
  1048. </div>
  1049. </div>
  1050. </div>
  1051. <div>
  1052. <label className="block text-sm text-bambu-gray mb-1">
  1053. Default Printer
  1054. </label>
  1055. <div className="relative">
  1056. <select
  1057. value={localSettings.default_printer_id ?? ''}
  1058. onChange={(e) => updateSetting('default_printer_id', e.target.value ? Number(e.target.value) : null)}
  1059. className="w-full px-3 py-2 pr-10 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none appearance-none cursor-pointer"
  1060. >
  1061. <option value="">{t('settings.noDefaultPrinter')}</option>
  1062. {printers?.map((printer) => (
  1063. <option key={printer.id} value={printer.id}>
  1064. {printer.name}
  1065. </option>
  1066. ))}
  1067. </select>
  1068. <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
  1069. </div>
  1070. <p className="text-xs text-bambu-gray mt-1">
  1071. Pre-select this printer for uploads, reprints, and other operations.
  1072. </p>
  1073. </div>
  1074. <div className="flex items-center justify-between">
  1075. <div>
  1076. <p className="text-white">{t('settings.sidebarOrder')}</p>
  1077. <p className="text-sm text-bambu-gray">
  1078. Drag items in the sidebar to reorder. Reset to default order here.
  1079. </p>
  1080. </div>
  1081. <Button
  1082. variant="secondary"
  1083. size="sm"
  1084. onClick={handleResetSidebarOrder}
  1085. >
  1086. <RotateCcw className="w-4 h-4" />
  1087. Reset
  1088. </Button>
  1089. </div>
  1090. </CardContent>
  1091. </Card>
  1092. <Card>
  1093. <CardHeader>
  1094. <h2 className="text-lg font-semibold text-white flex items-center gap-2">
  1095. <Palette className="w-5 h-5" />
  1096. Appearance
  1097. </h2>
  1098. </CardHeader>
  1099. <CardContent className="space-y-6">
  1100. {/* Dark Mode Settings */}
  1101. <div className={`space-y-3 p-4 rounded-lg border ${mode === 'dark' ? 'border-bambu-green bg-bambu-green/5' : 'border-bambu-dark-tertiary'}`}>
  1102. <h3 className="text-sm font-medium text-white flex items-center gap-2">
  1103. Dark Mode
  1104. {mode === 'dark' && <span className="text-xs text-bambu-green">(active)</span>}
  1105. </h3>
  1106. <div className="grid grid-cols-3 gap-3">
  1107. <div>
  1108. <label className="block text-xs text-bambu-gray mb-1">Background</label>
  1109. <select
  1110. value={darkBackground}
  1111. onChange={(e) => { setDarkBackground(e.target.value as DarkBackground); showToast(t('settings.toast.settingsSaved'), 'success'); }}
  1112. className="w-full px-2 py-1.5 text-sm bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  1113. >
  1114. <option value="neutral">Neutral</option>
  1115. <option value="warm">Warm</option>
  1116. <option value="cool">Cool</option>
  1117. <option value="oled">OLED Black</option>
  1118. <option value="slate">Slate Blue</option>
  1119. <option value="forest">Forest Green</option>
  1120. </select>
  1121. </div>
  1122. <div>
  1123. <label className="block text-xs text-bambu-gray mb-1">Accent</label>
  1124. <select
  1125. value={darkAccent}
  1126. onChange={(e) => { setDarkAccent(e.target.value as ThemeAccent); showToast(t('settings.toast.settingsSaved'), 'success'); }}
  1127. className="w-full px-2 py-1.5 text-sm bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  1128. >
  1129. <option value="green">Green</option>
  1130. <option value="teal">Teal</option>
  1131. <option value="blue">Blue</option>
  1132. <option value="orange">Orange</option>
  1133. <option value="purple">Purple</option>
  1134. <option value="red">Red</option>
  1135. </select>
  1136. </div>
  1137. <div>
  1138. <label className="block text-xs text-bambu-gray mb-1">Style</label>
  1139. <select
  1140. value={darkStyle}
  1141. onChange={(e) => { setDarkStyle(e.target.value as ThemeStyle); showToast(t('settings.toast.settingsSaved'), 'success'); }}
  1142. className="w-full px-2 py-1.5 text-sm bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  1143. >
  1144. <option value="classic">Classic</option>
  1145. <option value="glow">Glow</option>
  1146. <option value="vibrant">Vibrant</option>
  1147. </select>
  1148. </div>
  1149. </div>
  1150. </div>
  1151. {/* Light Mode Settings */}
  1152. <div className={`space-y-3 p-4 rounded-lg border ${mode === 'light' ? 'border-bambu-green bg-bambu-green/5' : 'border-bambu-dark-tertiary'}`}>
  1153. <h3 className="text-sm font-medium text-white flex items-center gap-2">
  1154. Light Mode
  1155. {mode === 'light' && <span className="text-xs text-bambu-green">(active)</span>}
  1156. </h3>
  1157. <div className="grid grid-cols-3 gap-3">
  1158. <div>
  1159. <label className="block text-xs text-bambu-gray mb-1">Background</label>
  1160. <select
  1161. value={lightBackground}
  1162. onChange={(e) => { setLightBackground(e.target.value as LightBackground); showToast(t('settings.toast.settingsSaved'), 'success'); }}
  1163. className="w-full px-2 py-1.5 text-sm bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  1164. >
  1165. <option value="neutral">Neutral</option>
  1166. <option value="warm">Warm</option>
  1167. <option value="cool">Cool</option>
  1168. </select>
  1169. </div>
  1170. <div>
  1171. <label className="block text-xs text-bambu-gray mb-1">Accent</label>
  1172. <select
  1173. value={lightAccent}
  1174. onChange={(e) => { setLightAccent(e.target.value as ThemeAccent); showToast(t('settings.toast.settingsSaved'), 'success'); }}
  1175. className="w-full px-2 py-1.5 text-sm bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  1176. >
  1177. <option value="green">Green</option>
  1178. <option value="teal">Teal</option>
  1179. <option value="blue">Blue</option>
  1180. <option value="orange">Orange</option>
  1181. <option value="purple">Purple</option>
  1182. <option value="red">Red</option>
  1183. </select>
  1184. </div>
  1185. <div>
  1186. <label className="block text-xs text-bambu-gray mb-1">Style</label>
  1187. <select
  1188. value={lightStyle}
  1189. onChange={(e) => { setLightStyle(e.target.value as ThemeStyle); showToast(t('settings.toast.settingsSaved'), 'success'); }}
  1190. className="w-full px-2 py-1.5 text-sm bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  1191. >
  1192. <option value="classic">Classic</option>
  1193. <option value="glow">Glow</option>
  1194. <option value="vibrant">Vibrant</option>
  1195. </select>
  1196. </div>
  1197. </div>
  1198. </div>
  1199. <p className="text-xs text-bambu-gray">
  1200. Toggle between dark and light mode using the sun/moon icon in the sidebar.
  1201. </p>
  1202. </CardContent>
  1203. </Card>
  1204. <Card>
  1205. <CardHeader>
  1206. <h2 className="text-lg font-semibold text-white">{t('settings.archiveSettings')}</h2>
  1207. </CardHeader>
  1208. <CardContent className="space-y-4">
  1209. <div className="flex items-center justify-between">
  1210. <div>
  1211. <p className="text-white">Auto-archive prints</p>
  1212. <p className="text-sm text-bambu-gray">
  1213. Automatically save 3MF files when prints complete
  1214. </p>
  1215. </div>
  1216. <label className="relative inline-flex items-center cursor-pointer">
  1217. <input
  1218. type="checkbox"
  1219. checked={localSettings.auto_archive}
  1220. onChange={(e) => updateSetting('auto_archive', e.target.checked)}
  1221. className="sr-only peer"
  1222. />
  1223. <div className="w-11 h-6 bg-bambu-dark-tertiary peer-focus:outline-none rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-bambu-green"></div>
  1224. </label>
  1225. </div>
  1226. <div className="flex items-center justify-between">
  1227. <div>
  1228. <p className="text-white">{t('settings.saveThumbnails')}</p>
  1229. <p className="text-sm text-bambu-gray">
  1230. Extract and save preview images from 3MF files
  1231. </p>
  1232. </div>
  1233. <label className="relative inline-flex items-center cursor-pointer">
  1234. <input
  1235. type="checkbox"
  1236. checked={localSettings.save_thumbnails}
  1237. onChange={(e) => updateSetting('save_thumbnails', e.target.checked)}
  1238. className="sr-only peer"
  1239. />
  1240. <div className="w-11 h-6 bg-bambu-dark-tertiary peer-focus:outline-none rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-bambu-green"></div>
  1241. </label>
  1242. </div>
  1243. <div className="flex items-center justify-between">
  1244. <div>
  1245. <p className="text-white">{t('settings.captureFinishPhoto')}</p>
  1246. <p className="text-sm text-bambu-gray">
  1247. Take a photo from printer camera when print completes
  1248. </p>
  1249. </div>
  1250. <label className="relative inline-flex items-center cursor-pointer">
  1251. <input
  1252. type="checkbox"
  1253. checked={localSettings.capture_finish_photo}
  1254. onChange={(e) => updateSetting('capture_finish_photo', e.target.checked)}
  1255. className="sr-only peer"
  1256. />
  1257. <div className="w-11 h-6 bg-bambu-dark-tertiary peer-focus:outline-none rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-bambu-green"></div>
  1258. </label>
  1259. </div>
  1260. {localSettings.capture_finish_photo && ffmpegStatus && !ffmpegStatus.installed && (
  1261. <div className="flex items-start gap-2 p-3 bg-yellow-500/10 border border-yellow-500/30 rounded-lg">
  1262. <AlertTriangle className="w-5 h-5 text-yellow-500 flex-shrink-0 mt-0.5" />
  1263. <div className="text-sm">
  1264. <p className="text-yellow-500 font-medium">ffmpeg not installed</p>
  1265. <p className="text-bambu-gray mt-1">
  1266. Camera capture requires ffmpeg. Install it via{' '}
  1267. <code className="bg-bambu-dark-tertiary px-1 rounded">brew install ffmpeg</code> (macOS) or{' '}
  1268. <code className="bg-bambu-dark-tertiary px-1 rounded">apt install ffmpeg</code> (Linux).
  1269. </p>
  1270. </div>
  1271. </div>
  1272. )}
  1273. </CardContent>
  1274. </Card>
  1275. </div>
  1276. {/* Second Column - Camera, Cost, AMS & Spoolman */}
  1277. <div className="space-y-6 flex-1 lg:max-w-md">
  1278. {/* Camera Settings */}
  1279. <Card>
  1280. <CardHeader>
  1281. <h2 className="text-lg font-semibold text-white flex items-center gap-2">
  1282. <Video className="w-5 h-5 text-bambu-green" />
  1283. Camera
  1284. </h2>
  1285. </CardHeader>
  1286. <CardContent className="space-y-4">
  1287. <div>
  1288. <label className="block text-sm text-bambu-gray mb-1">
  1289. Camera View Mode
  1290. </label>
  1291. <select
  1292. value={localSettings.camera_view_mode ?? 'window'}
  1293. onChange={(e) => updateSetting('camera_view_mode', e.target.value as 'window' | 'embedded')}
  1294. 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"
  1295. >
  1296. <option value="window">{t('settings.newWindow')}</option>
  1297. <option value="embedded">{t('settings.embeddedOverlay')}</option>
  1298. </select>
  1299. <p className="text-xs text-bambu-gray mt-1">
  1300. {localSettings.camera_view_mode === 'embedded'
  1301. ? 'Camera opens in a resizable overlay on the main screen'
  1302. : 'Camera opens in a separate browser window'}
  1303. </p>
  1304. </div>
  1305. {/* External Cameras Section */}
  1306. <div className="border-t border-bambu-dark-tertiary pt-4 mt-4">
  1307. <h3 className="text-sm font-medium text-white mb-2">{t('settings.externalCameras')}</h3>
  1308. <p className="text-xs text-bambu-gray mb-3">
  1309. Configure external cameras to replace the built-in printer camera. Supports MJPEG streams, RTSP, HTTP snapshots, and USB cameras (V4L2). When enabled, the external camera is used for live view and finish photos.
  1310. </p>
  1311. {printers && printers.length > 0 ? (
  1312. <div className="space-y-3">
  1313. {printers.map(printer => (
  1314. <div key={printer.id} className="p-3 bg-bambu-dark rounded-lg">
  1315. <div className="flex items-center justify-between mb-2">
  1316. <span className="text-white font-medium text-sm">{printer.name}</span>
  1317. <label className="relative inline-flex items-center cursor-pointer">
  1318. <input
  1319. type="checkbox"
  1320. checked={printer.external_camera_enabled}
  1321. onChange={(e) => handleUpdatePrinterCamera(printer.id, { enabled: e.target.checked })}
  1322. className="sr-only peer"
  1323. />
  1324. <div className="w-9 h-5 bg-bambu-dark-tertiary peer-focus:outline-none rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:border-gray-300 after:border after:rounded-full after:h-4 after:w-4 after:transition-all peer-checked:bg-bambu-green"></div>
  1325. </label>
  1326. </div>
  1327. {printer.external_camera_enabled && (
  1328. <div className="space-y-2 mt-2">
  1329. <input
  1330. type="text"
  1331. placeholder={printer.external_camera_type === 'usb' ? 'Device path (/dev/video0)' : 'Camera URL (rtsp://... or http://...)'}
  1332. value={localCameraUrls[printer.id] ?? printer.external_camera_url ?? ''}
  1333. onChange={(e) => handleCameraUrlChange(printer.id, e.target.value)}
  1334. className="w-full px-3 py-2 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded text-white text-sm focus:border-bambu-green focus:outline-none"
  1335. />
  1336. <div className="flex gap-2">
  1337. <select
  1338. value={printer.external_camera_type || 'mjpeg'}
  1339. onChange={(e) => handleUpdatePrinterCamera(printer.id, { type: e.target.value })}
  1340. className="flex-1 px-3 py-2 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded text-white text-sm focus:border-bambu-green focus:outline-none"
  1341. >
  1342. <option value="mjpeg">MJPEG Stream</option>
  1343. <option value="rtsp">RTSP Stream</option>
  1344. <option value="snapshot">HTTP Snapshot</option>
  1345. <option value="usb">USB Camera (V4L2)</option>
  1346. </select>
  1347. <Button
  1348. size="sm"
  1349. variant="secondary"
  1350. onClick={() => handleTestExternalCamera(printer.id, localCameraUrls[printer.id] ?? printer.external_camera_url ?? '', printer.external_camera_type || 'mjpeg')}
  1351. disabled={extCameraTestLoading[printer.id] || !(localCameraUrls[printer.id] ?? printer.external_camera_url)}
  1352. >
  1353. {extCameraTestLoading[printer.id] ? (
  1354. <Loader2 className="w-4 h-4 animate-spin" />
  1355. ) : (
  1356. 'Test'
  1357. )}
  1358. </Button>
  1359. </div>
  1360. {extCameraTestResults[printer.id] && (
  1361. <div className={`text-xs flex items-center gap-1 ${extCameraTestResults[printer.id]?.success ? 'text-green-500' : 'text-red-500'}`}>
  1362. {extCameraTestResults[printer.id]?.success ? (
  1363. <>
  1364. <CheckCircle className="w-3 h-3" />
  1365. Connected{extCameraTestResults[printer.id]?.resolution && ` (${extCameraTestResults[printer.id]?.resolution})`}
  1366. </>
  1367. ) : (
  1368. <>
  1369. <XCircle className="w-3 h-3" />
  1370. {extCameraTestResults[printer.id]?.error || t('settings.toast.connectionFailed')}
  1371. </>
  1372. )}
  1373. </div>
  1374. )}
  1375. </div>
  1376. )}
  1377. </div>
  1378. ))}
  1379. </div>
  1380. ) : (
  1381. <p className="text-xs text-bambu-gray italic">{t('settings.noPrintersConfigured')}</p>
  1382. )}
  1383. </div>
  1384. </CardContent>
  1385. </Card>
  1386. <Card>
  1387. <CardHeader>
  1388. <h2 className="text-lg font-semibold text-white">{t('settings.costTracking')}</h2>
  1389. </CardHeader>
  1390. <CardContent className="space-y-4">
  1391. <div>
  1392. <label className="block text-sm text-bambu-gray mb-1">
  1393. Default filament cost (per kg)
  1394. </label>
  1395. <input
  1396. type="number"
  1397. step="0.01"
  1398. min="0"
  1399. value={localSettings.default_filament_cost}
  1400. onChange={(e) =>
  1401. updateSetting('default_filament_cost', parseFloat(e.target.value) || 0)
  1402. }
  1403. 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"
  1404. />
  1405. </div>
  1406. <div>
  1407. <label className="block text-sm text-bambu-gray mb-1">Currency</label>
  1408. <select
  1409. value={localSettings.currency}
  1410. onChange={(e) => updateSetting('currency', e.target.value)}
  1411. 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"
  1412. >
  1413. <option value="USD">USD ($)</option>
  1414. <option value="EUR">EUR (€)</option>
  1415. <option value="GBP">GBP (£)</option>
  1416. <option value="CHF">CHF (Fr.)</option>
  1417. <option value="JPY">JPY (¥)</option>
  1418. <option value="CNY">CNY (¥)</option>
  1419. <option value="CAD">CAD ($)</option>
  1420. <option value="AUD">AUD ($)</option>
  1421. </select>
  1422. </div>
  1423. <div>
  1424. <label className="block text-sm text-bambu-gray mb-1">
  1425. Electricity cost per kWh
  1426. </label>
  1427. <input
  1428. type="number"
  1429. step="0.01"
  1430. min="0"
  1431. value={localSettings.energy_cost_per_kwh}
  1432. onChange={(e) =>
  1433. updateSetting('energy_cost_per_kwh', parseFloat(e.target.value) || 0)
  1434. }
  1435. 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"
  1436. />
  1437. </div>
  1438. <div>
  1439. <label className="block text-sm text-bambu-gray mb-1">
  1440. Energy display mode
  1441. </label>
  1442. <select
  1443. value={localSettings.energy_tracking_mode || 'total'}
  1444. onChange={(e) => updateSetting('energy_tracking_mode', e.target.value as 'print' | 'total')}
  1445. 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"
  1446. >
  1447. <option value="print">{t('settings.printsOnly')}</option>
  1448. <option value="total">{t('settings.totalConsumption')}</option>
  1449. </select>
  1450. <p className="text-xs text-bambu-gray mt-1">
  1451. {localSettings.energy_tracking_mode === 'print'
  1452. ? 'Dashboard shows sum of energy used during prints'
  1453. : 'Dashboard shows lifetime energy from smart plugs'}
  1454. </p>
  1455. </div>
  1456. </CardContent>
  1457. </Card>
  1458. {/* File Manager Settings */}
  1459. <Card>
  1460. <CardHeader>
  1461. <h2 className="text-lg font-semibold text-white flex items-center gap-2">
  1462. <FileText className="w-5 h-5 text-bambu-green" />
  1463. File Manager
  1464. </h2>
  1465. </CardHeader>
  1466. <CardContent className="space-y-4">
  1467. {/* Archive Mode */}
  1468. <div>
  1469. <label className="block text-sm text-bambu-gray mb-1">
  1470. Create Archive Entry When Printing
  1471. </label>
  1472. <select
  1473. value={localSettings.library_archive_mode ?? 'ask'}
  1474. onChange={(e) => updateSetting('library_archive_mode', e.target.value as 'always' | 'never' | 'ask')}
  1475. 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"
  1476. >
  1477. <option value="always">{t('settings.archiveMode.always')}</option>
  1478. <option value="never">{t('settings.archiveMode.never')}</option>
  1479. <option value="ask">{t('settings.archiveMode.ask')}</option>
  1480. </select>
  1481. <p className="text-xs text-bambu-gray mt-1">
  1482. When printing from File Manager, optionally create an archive entry
  1483. </p>
  1484. </div>
  1485. {/* Disk Space Warning Threshold */}
  1486. <div>
  1487. <label className="block text-sm text-bambu-gray mb-1">
  1488. Low Disk Space Warning
  1489. </label>
  1490. <div className="flex items-center gap-2">
  1491. <input
  1492. type="number"
  1493. min="0.5"
  1494. max="100"
  1495. step="0.5"
  1496. value={localSettings.library_disk_warning_gb ?? 5}
  1497. onChange={(e) => updateSetting('library_disk_warning_gb', parseFloat(e.target.value) || 5)}
  1498. className="w-24 px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  1499. />
  1500. <span className="text-bambu-gray">GB</span>
  1501. </div>
  1502. <p className="text-xs text-bambu-gray mt-1">
  1503. Show warning when free disk space falls below this threshold
  1504. </p>
  1505. </div>
  1506. </CardContent>
  1507. </Card>
  1508. </div>
  1509. {/* Third Column - Sidebar Links & Updates */}
  1510. <div className="space-y-6 flex-1 lg:max-w-sm">
  1511. {/* Sidebar Links */}
  1512. <ExternalLinksSettings />
  1513. <Card>
  1514. <CardHeader>
  1515. <h2 className="text-lg font-semibold text-white">Updates</h2>
  1516. </CardHeader>
  1517. <CardContent className="space-y-4">
  1518. <div className="flex items-center justify-between">
  1519. <div>
  1520. <p className="text-white">{t('settings.checkForUpdatesLabel')}</p>
  1521. <p className="text-sm text-bambu-gray">
  1522. Automatically check for new versions on startup
  1523. </p>
  1524. </div>
  1525. <label className="relative inline-flex items-center cursor-pointer">
  1526. <input
  1527. type="checkbox"
  1528. checked={localSettings.check_updates}
  1529. onChange={(e) => updateSetting('check_updates', e.target.checked)}
  1530. className="sr-only peer"
  1531. />
  1532. <div className="w-11 h-6 bg-bambu-dark-tertiary peer-focus:outline-none rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-bambu-green"></div>
  1533. </label>
  1534. </div>
  1535. <div className="flex items-center justify-between">
  1536. <div>
  1537. <p className="text-white">{t('settings.checkPrinterFirmware')}</p>
  1538. <p className="text-sm text-bambu-gray">
  1539. Check for printer firmware updates from Bambu Lab
  1540. </p>
  1541. </div>
  1542. <label className="relative inline-flex items-center cursor-pointer">
  1543. <input
  1544. type="checkbox"
  1545. checked={localSettings.check_printer_firmware ?? true}
  1546. onChange={(e) => updateSetting('check_printer_firmware', e.target.checked)}
  1547. className="sr-only peer"
  1548. />
  1549. <div className="w-11 h-6 bg-bambu-dark-tertiary peer-focus:outline-none rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-bambu-green"></div>
  1550. </label>
  1551. </div>
  1552. <div className="border-t border-bambu-dark-tertiary pt-4">
  1553. <div className="flex items-center justify-between mb-2">
  1554. <div>
  1555. <p className="text-white">{t('settings.currentVersion')}</p>
  1556. <p className="text-sm text-bambu-gray">v{versionInfo?.version || '...'}</p>
  1557. </div>
  1558. <Button
  1559. variant="secondary"
  1560. size="sm"
  1561. onClick={() => refetchUpdateCheck()}
  1562. disabled={isCheckingUpdate}
  1563. >
  1564. {isCheckingUpdate ? (
  1565. <Loader2 className="w-4 h-4 animate-spin" />
  1566. ) : (
  1567. <RefreshCw className="w-4 h-4" />
  1568. )}
  1569. Check now
  1570. </Button>
  1571. </div>
  1572. {updateCheck?.update_available ? (
  1573. <div className="mt-4 p-3 bg-bambu-green/10 border border-bambu-green/30 rounded-lg">
  1574. <div className="flex items-start justify-between">
  1575. <div>
  1576. <p className="text-bambu-green font-medium">
  1577. Update available: v{updateCheck.latest_version}
  1578. </p>
  1579. {updateCheck.release_name && updateCheck.release_name !== updateCheck.latest_version && (
  1580. <p className="text-sm text-bambu-gray mt-1">{updateCheck.release_name}</p>
  1581. )}
  1582. </div>
  1583. <div className="flex items-center gap-2">
  1584. {updateCheck.release_notes && (
  1585. <button
  1586. onClick={() => setShowReleaseNotes(true)}
  1587. className="text-bambu-gray hover:text-white transition-colors text-sm underline"
  1588. >
  1589. Release Notes
  1590. </button>
  1591. )}
  1592. {updateCheck.release_url && (
  1593. <a
  1594. href={updateCheck.release_url}
  1595. target="_blank"
  1596. rel="noopener noreferrer"
  1597. className="text-bambu-gray hover:text-white transition-colors"
  1598. title={t('settings.viewReleaseOnGitHub')}
  1599. >
  1600. <ExternalLink className="w-4 h-4" />
  1601. </a>
  1602. )}
  1603. </div>
  1604. </div>
  1605. {updateStatus?.status === 'downloading' || updateStatus?.status === 'installing' ? (
  1606. <div className="mt-3">
  1607. <div className="flex items-center gap-2 text-sm text-bambu-gray">
  1608. <Loader2 className="w-4 h-4 animate-spin" />
  1609. <span>{updateStatus.message}</span>
  1610. </div>
  1611. <div className="mt-2 w-full bg-bambu-dark-tertiary rounded-full h-2">
  1612. <div
  1613. className="bg-bambu-green h-2 rounded-full transition-all duration-300"
  1614. style={{ width: `${updateStatus.progress}%` }}
  1615. />
  1616. </div>
  1617. </div>
  1618. ) : updateStatus?.status === 'complete' ? (
  1619. <div className="mt-3 p-2 bg-bambu-green/20 rounded text-sm text-bambu-green">
  1620. {updateStatus.message}
  1621. </div>
  1622. ) : updateStatus?.status === 'error' ? (
  1623. <div className="mt-3 p-2 bg-red-500/20 rounded text-sm text-red-400">
  1624. {updateStatus.error || updateStatus.message}
  1625. </div>
  1626. ) : updateCheck?.is_docker ? (
  1627. <div className="mt-3 p-3 bg-bambu-dark-tertiary rounded-lg">
  1628. <p className="text-sm text-bambu-gray mb-2">
  1629. Update via Docker Compose:
  1630. </p>
  1631. <code className="block text-xs bg-bambu-dark p-2 rounded text-bambu-green font-mono">
  1632. docker compose pull && docker compose up -d
  1633. </code>
  1634. </div>
  1635. ) : (
  1636. <Button
  1637. className="mt-3"
  1638. onClick={() => applyUpdateMutation.mutate()}
  1639. disabled={applyUpdateMutation.isPending}
  1640. >
  1641. {applyUpdateMutation.isPending ? (
  1642. <Loader2 className="w-4 h-4 animate-spin" />
  1643. ) : (
  1644. <Download className="w-4 h-4" />
  1645. )}
  1646. Install Update
  1647. </Button>
  1648. )}
  1649. </div>
  1650. ) : updateCheck?.error ? (
  1651. <div className="mt-2 p-2 bg-red-500/10 border border-red-500/30 rounded text-sm text-red-400">
  1652. Failed to check for updates: {updateCheck.error}
  1653. </div>
  1654. ) : updateCheck && !updateCheck.update_available ? (
  1655. <p className="mt-2 text-sm text-bambu-gray">
  1656. You're running the latest version
  1657. </p>
  1658. ) : null}
  1659. </div>
  1660. </CardContent>
  1661. </Card>
  1662. {/* Data Management */}
  1663. <Card>
  1664. <CardHeader>
  1665. <h2 className="text-lg font-semibold text-white">{t('settings.dataManagement')}</h2>
  1666. </CardHeader>
  1667. <CardContent className="space-y-4">
  1668. <div className="flex items-center justify-between">
  1669. <div>
  1670. <p className="text-white">{t('settings.clearNotificationLogs')}</p>
  1671. <p className="text-sm text-bambu-gray">
  1672. {t('settings.clearNotificationLogsDescription')}
  1673. </p>
  1674. </div>
  1675. <Button
  1676. variant="secondary"
  1677. size="sm"
  1678. onClick={() => setShowClearLogsConfirm(true)}
  1679. >
  1680. <Trash2 className="w-4 h-4" />
  1681. {t('common.clear')}
  1682. </Button>
  1683. </div>
  1684. <div className="flex items-center justify-between">
  1685. <div>
  1686. <p className="text-white">{t('settings.resetUiPreferences')}</p>
  1687. <p className="text-sm text-bambu-gray">
  1688. {t('settings.resetUiPreferencesDescription')}
  1689. </p>
  1690. </div>
  1691. <Button
  1692. variant="secondary"
  1693. size="sm"
  1694. onClick={() => setShowClearStorageConfirm(true)}
  1695. >
  1696. <Trash2 className="w-4 h-4" />
  1697. Reset
  1698. </Button>
  1699. </div>
  1700. <div className="flex items-center justify-between pt-4 border-t border-bambu-dark-tertiary">
  1701. <div>
  1702. <p className="text-white">Backup & Restore</p>
  1703. <p className="text-sm text-bambu-gray">
  1704. Export/import settings and configure GitHub backup
  1705. </p>
  1706. </div>
  1707. <Button
  1708. variant="secondary"
  1709. size="sm"
  1710. onClick={() => handleTabChange('backup')}
  1711. >
  1712. <Database className="w-4 h-4" />
  1713. Go to Backup
  1714. </Button>
  1715. </div>
  1716. </CardContent>
  1717. </Card>
  1718. </div>
  1719. </div>
  1720. )}
  1721. {/* Network Tab */}
  1722. {activeTab === 'network' && localSettings && (
  1723. <div className="flex flex-col lg:flex-row gap-6">
  1724. {/* Left Column - External URL & FTP Retry */}
  1725. <div className="flex-1 lg:max-w-xl space-y-4">
  1726. {/* External URL */}
  1727. <Card>
  1728. <CardHeader>
  1729. <h2 className="text-lg font-semibold text-white flex items-center gap-2">
  1730. <Globe className="w-5 h-5 text-blue-400" />
  1731. External URL
  1732. </h2>
  1733. </CardHeader>
  1734. <CardContent className="space-y-4">
  1735. <p className="text-sm text-bambu-gray">
  1736. The external URL where Bambuddy is accessible. Used for notification images and external integrations.
  1737. </p>
  1738. <div>
  1739. <label className="block text-sm text-bambu-gray mb-1">
  1740. Bambuddy URL
  1741. </label>
  1742. <input
  1743. type="text"
  1744. value={localSettings.external_url ?? ''}
  1745. onChange={(e) => updateSetting('external_url', e.target.value)}
  1746. placeholder="http://192.168.1.100:8000"
  1747. 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"
  1748. />
  1749. <p className="text-xs text-bambu-gray mt-1">
  1750. Include protocol and port (e.g., http://192.168.1.100:8000)
  1751. </p>
  1752. </div>
  1753. </CardContent>
  1754. </Card>
  1755. <Card>
  1756. <CardHeader>
  1757. <h2 className="text-lg font-semibold text-white flex items-center gap-2">
  1758. <RefreshCw className="w-5 h-5 text-blue-400" />
  1759. FTP Retry
  1760. </h2>
  1761. </CardHeader>
  1762. <CardContent className="space-y-4">
  1763. <p className="text-sm text-bambu-gray">
  1764. Retry FTP operations when printer WiFi is unreliable. Applies to 3MF downloads, print uploads, timelapse downloads, and firmware updates.
  1765. </p>
  1766. <div className="flex items-center justify-between">
  1767. <div>
  1768. <p className="text-white">{t('settings.enableRetry')}</p>
  1769. <p className="text-sm text-bambu-gray">
  1770. Automatically retry failed FTP operations
  1771. </p>
  1772. </div>
  1773. <label className="relative inline-flex items-center cursor-pointer">
  1774. <input
  1775. type="checkbox"
  1776. checked={localSettings.ftp_retry_enabled ?? true}
  1777. onChange={(e) => updateSetting('ftp_retry_enabled', e.target.checked)}
  1778. className="sr-only peer"
  1779. />
  1780. <div className="w-11 h-6 bg-bambu-dark-tertiary peer-focus:outline-none rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-bambu-green"></div>
  1781. </label>
  1782. </div>
  1783. {localSettings.ftp_retry_enabled && (
  1784. <div className="space-y-4 pt-2 border-t border-bambu-dark-tertiary">
  1785. <div>
  1786. <label className="block text-sm text-bambu-gray mb-1">
  1787. Retry attempts
  1788. </label>
  1789. <div className="relative w-44">
  1790. <select
  1791. value={localSettings.ftp_retry_count ?? 3}
  1792. onChange={(e) => updateSetting('ftp_retry_count', parseInt(e.target.value))}
  1793. className="w-full px-3 py-2 pr-10 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none appearance-none cursor-pointer"
  1794. >
  1795. {[1, 2, 3, 4, 5, 6, 7, 8, 9, 10].map(n => (
  1796. <option key={n} value={n}>{n} {n === 1 ? 'time' : 'times'}</option>
  1797. ))}
  1798. </select>
  1799. <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
  1800. </div>
  1801. </div>
  1802. <div>
  1803. <label className="block text-sm text-bambu-gray mb-1">
  1804. Retry delay
  1805. </label>
  1806. <div className="relative w-44">
  1807. <select
  1808. value={localSettings.ftp_retry_delay ?? 2}
  1809. onChange={(e) => updateSetting('ftp_retry_delay', parseInt(e.target.value))}
  1810. className="w-full px-3 py-2 pr-10 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none appearance-none cursor-pointer"
  1811. >
  1812. {[1, 2, 3, 5, 10, 15, 20, 30].map(n => (
  1813. <option key={n} value={n}>{n} {n === 1 ? 'second' : 'seconds'}</option>
  1814. ))}
  1815. </select>
  1816. <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
  1817. </div>
  1818. </div>
  1819. <div>
  1820. <label className="block text-sm text-bambu-gray mb-1">
  1821. Connection timeout
  1822. </label>
  1823. <div className="relative w-44">
  1824. <select
  1825. value={localSettings.ftp_timeout ?? 30}
  1826. onChange={(e) => updateSetting('ftp_timeout', parseInt(e.target.value))}
  1827. className="w-full px-3 py-2 pr-10 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none appearance-none cursor-pointer"
  1828. >
  1829. {[10, 15, 20, 30, 45, 60, 90, 120].map(n => (
  1830. <option key={n} value={n}>{n} seconds</option>
  1831. ))}
  1832. </select>
  1833. <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
  1834. </div>
  1835. <p className="text-xs text-bambu-gray mt-1">
  1836. Increase for printers with weak WiFi
  1837. </p>
  1838. </div>
  1839. </div>
  1840. )}
  1841. </CardContent>
  1842. </Card>
  1843. </div>
  1844. {/* Right Column - Home Assistant & MQTT Publishing */}
  1845. <div className="flex-1 lg:max-w-xl space-y-4">
  1846. {/* Home Assistant Integration */}
  1847. <Card>
  1848. <CardHeader>
  1849. <div className="flex items-center justify-between">
  1850. <h2 className="text-lg font-semibold text-white flex items-center gap-2">
  1851. <Home className="w-5 h-5 text-bambu-green" />
  1852. Home Assistant
  1853. </h2>
  1854. {localSettings.ha_enabled && haTestResult && (
  1855. <div className="flex items-center gap-2">
  1856. <span className={`w-2.5 h-2.5 rounded-full ${haTestResult.success ? 'bg-green-400' : 'bg-red-400'}`} />
  1857. <span className={`text-sm ${haTestResult.success ? 'text-green-400' : 'text-red-400'}`}>
  1858. {haTestResult.success ? 'Connected' : 'Disconnected'}
  1859. </span>
  1860. </div>
  1861. )}
  1862. </div>
  1863. </CardHeader>
  1864. <CardContent className="space-y-4">
  1865. <p className="text-sm text-bambu-gray">
  1866. Connect to Home Assistant to control smart plugs via HA's REST API. Supports switch, light, input_boolean, and script entities.
  1867. </p>
  1868. <div className="flex items-center justify-between">
  1869. <div className="flex-1">
  1870. <p className="text-white">{t('settings.enableHomeAssistant')}</p>
  1871. <p className="text-xs text-bambu-gray">{t('settings.homeAssistantDescription')}</p>
  1872. {localSettings.ha_env_managed && (
  1873. <div className="flex items-center gap-1 mt-1">
  1874. <Lock className="w-3 h-3 text-bambu-green" />
  1875. <span className="text-xs text-bambu-green">
  1876. {t('settings.autoEnabledViaEnv')}
  1877. </span>
  1878. </div>
  1879. )}
  1880. </div>
  1881. <label className="relative inline-flex items-center cursor-pointer">
  1882. <input
  1883. type="checkbox"
  1884. checked={localSettings.ha_enabled ?? false}
  1885. onChange={(e) => updateSetting('ha_enabled', e.target.checked)}
  1886. disabled={localSettings.ha_env_managed}
  1887. className="sr-only peer"
  1888. />
  1889. <div className={`w-11 h-6 bg-bambu-dark-tertiary peer-focus:outline-none rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-bambu-green ${
  1890. localSettings.ha_env_managed ? 'opacity-60 cursor-not-allowed' : ''
  1891. }`}></div>
  1892. </label>
  1893. </div>
  1894. {localSettings.ha_enabled && (
  1895. <>
  1896. <div>
  1897. <label className="block text-sm text-bambu-gray mb-1">
  1898. Home Assistant URL
  1899. {localSettings.ha_url_from_env && (
  1900. <span className="ml-2 text-xs text-bambu-green">
  1901. {t('settings.environmentManagedLabel')}
  1902. </span>
  1903. )}
  1904. </label>
  1905. <div className="relative">
  1906. <input
  1907. type="text"
  1908. value={localSettings.ha_url ?? ''}
  1909. onChange={(e) => updateSetting('ha_url', e.target.value)}
  1910. placeholder="http://192.168.1.100:8123"
  1911. disabled={localSettings.ha_url_from_env}
  1912. 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 ${
  1913. localSettings.ha_url_from_env ? 'opacity-60 cursor-not-allowed' : ''
  1914. }`}
  1915. />
  1916. {localSettings.ha_url_from_env && (
  1917. <Lock className="absolute right-3 top-2.5 w-4 h-4 text-bambu-gray" />
  1918. )}
  1919. </div>
  1920. {localSettings.ha_url_from_env && (
  1921. <p className="text-xs text-bambu-gray mt-1">
  1922. {t('settings.urlFromEnvReadOnly')}
  1923. </p>
  1924. )}
  1925. </div>
  1926. <div>
  1927. <label className="block text-sm text-bambu-gray mb-1">
  1928. Long-Lived Access Token
  1929. {localSettings.ha_token_from_env && (
  1930. <span className="ml-2 text-xs text-bambu-green">
  1931. {t('settings.environmentManagedLabel')}
  1932. </span>
  1933. )}
  1934. </label>
  1935. <div className="relative">
  1936. <input
  1937. type="password"
  1938. value={localSettings.ha_token ?? ''}
  1939. onChange={(e) => updateSetting('ha_token', e.target.value)}
  1940. placeholder="eyJ0eXAiOiJKV1QiLC..."
  1941. disabled={localSettings.ha_token_from_env}
  1942. 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 ${
  1943. localSettings.ha_token_from_env ? 'opacity-60 cursor-not-allowed' : ''
  1944. }`}
  1945. />
  1946. {localSettings.ha_token_from_env && (
  1947. <Lock className="absolute right-3 top-2.5 w-4 h-4 text-bambu-gray" />
  1948. )}
  1949. </div>
  1950. {localSettings.ha_token_from_env ? (
  1951. <p className="text-xs text-bambu-gray mt-1">
  1952. {t('settings.tokenFromEnvReadOnly')}
  1953. </p>
  1954. ) : (
  1955. <p className="text-xs text-bambu-gray mt-1">
  1956. Create a token in HA: Profile → Long-Lived Access Tokens → Create Token
  1957. </p>
  1958. )}
  1959. </div>
  1960. {localSettings.ha_url && localSettings.ha_token && (
  1961. <div className="pt-2 border-t border-bambu-dark-tertiary">
  1962. <Button
  1963. variant="secondary"
  1964. size="sm"
  1965. disabled={haTestLoading}
  1966. onClick={async () => {
  1967. setHaTestLoading(true);
  1968. setHaTestResult(null);
  1969. try {
  1970. const result = await api.testHAConnection(localSettings.ha_url!, localSettings.ha_token!);
  1971. setHaTestResult(result);
  1972. } catch (e) {
  1973. setHaTestResult({ success: false, message: null, error: e instanceof Error ? e.message : t('common.unknownError') });
  1974. } finally {
  1975. setHaTestLoading(false);
  1976. }
  1977. }}
  1978. >
  1979. {haTestLoading ? <Loader2 className="w-4 h-4 animate-spin" /> : <Wifi className="w-4 h-4" />}
  1980. {t('settings.testConnection')}
  1981. </Button>
  1982. </div>
  1983. )}
  1984. </>
  1985. )}
  1986. </CardContent>
  1987. </Card>
  1988. {/* MQTT Publishing */}
  1989. <Card>
  1990. <CardHeader>
  1991. <div className="flex items-center justify-between">
  1992. <h2 className="text-lg font-semibold text-white flex items-center gap-2">
  1993. <Wifi className="w-5 h-5 text-blue-400" />
  1994. MQTT Publishing
  1995. </h2>
  1996. {mqttStatus?.enabled && (
  1997. <div className="flex items-center gap-2">
  1998. <span className={`w-2.5 h-2.5 rounded-full ${mqttStatus.connected ? 'bg-green-400' : 'bg-red-400'}`} />
  1999. <span className={`text-sm ${mqttStatus.connected ? 'text-green-400' : 'text-red-400'}`}>
  2000. {mqttStatus.connected ? 'Connected' : 'Disconnected'}
  2001. </span>
  2002. </div>
  2003. )}
  2004. </div>
  2005. </CardHeader>
  2006. <CardContent className="space-y-4">
  2007. <p className="text-sm text-bambu-gray">
  2008. Publish BamBuddy events to an external MQTT broker for integration with Node-RED, Home Assistant, and other automation systems.
  2009. </p>
  2010. <div className="flex items-center justify-between">
  2011. <div>
  2012. <p className="text-white">{t('settings.enableMqtt')}</p>
  2013. <p className="text-sm text-bambu-gray">
  2014. Publish events to external MQTT broker
  2015. </p>
  2016. </div>
  2017. <label className="relative inline-flex items-center cursor-pointer">
  2018. <input
  2019. type="checkbox"
  2020. checked={localSettings.mqtt_enabled ?? false}
  2021. onChange={(e) => updateSetting('mqtt_enabled', e.target.checked)}
  2022. className="sr-only peer"
  2023. />
  2024. <div className="w-11 h-6 bg-bambu-dark-tertiary peer-focus:outline-none rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-bambu-green"></div>
  2025. </label>
  2026. </div>
  2027. {localSettings.mqtt_enabled && (
  2028. <div className="space-y-4 pt-2 border-t border-bambu-dark-tertiary">
  2029. <div>
  2030. <label className="block text-sm text-bambu-gray mb-1">
  2031. Broker hostname
  2032. </label>
  2033. <input
  2034. type="text"
  2035. value={localSettings.mqtt_broker ?? ''}
  2036. onChange={(e) => updateSetting('mqtt_broker', e.target.value)}
  2037. placeholder="mqtt.example.com or 192.168.1.100"
  2038. 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"
  2039. />
  2040. </div>
  2041. <div className="flex items-end gap-4">
  2042. <div className="flex-1">
  2043. <label className="block text-sm text-bambu-gray mb-1">
  2044. Port
  2045. </label>
  2046. <input
  2047. type="number"
  2048. min="1"
  2049. max="65535"
  2050. value={localSettings.mqtt_port ?? 1883}
  2051. onChange={(e) => updateSetting('mqtt_port', Math.min(65535, Math.max(1, parseInt(e.target.value) || 1883)))}
  2052. className="w-24 px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  2053. />
  2054. </div>
  2055. <div className="flex items-center gap-3 pb-2">
  2056. <label className="relative inline-flex items-center cursor-pointer">
  2057. <input
  2058. type="checkbox"
  2059. checked={localSettings.mqtt_use_tls ?? false}
  2060. onChange={(e) => {
  2061. const useTls = e.target.checked;
  2062. updateSetting('mqtt_use_tls', useTls);
  2063. // Auto-populate port based on TLS selection
  2064. const currentPort = localSettings.mqtt_port ?? 1883;
  2065. if (useTls && currentPort === 1883) {
  2066. updateSetting('mqtt_port', 8883);
  2067. } else if (!useTls && currentPort === 8883) {
  2068. updateSetting('mqtt_port', 1883);
  2069. }
  2070. }}
  2071. className="sr-only peer"
  2072. />
  2073. <div className="w-11 h-6 bg-bambu-dark-tertiary peer-focus:outline-none rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-bambu-green"></div>
  2074. </label>
  2075. <span className="text-white text-sm">{t('settings.useTls')}</span>
  2076. </div>
  2077. </div>
  2078. <div>
  2079. <label className="block text-sm text-bambu-gray mb-1">
  2080. Username (optional)
  2081. </label>
  2082. <input
  2083. type="text"
  2084. value={localSettings.mqtt_username ?? ''}
  2085. onChange={(e) => updateSetting('mqtt_username', e.target.value)}
  2086. placeholder={t('settings.leaveEmptyForAnonymous')}
  2087. 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"
  2088. />
  2089. </div>
  2090. <div>
  2091. <label className="block text-sm text-bambu-gray mb-1">
  2092. Password (optional)
  2093. </label>
  2094. <input
  2095. type="password"
  2096. value={localSettings.mqtt_password ?? ''}
  2097. onChange={(e) => updateSetting('mqtt_password', e.target.value)}
  2098. placeholder={t('settings.leaveEmptyForAnonymous')}
  2099. 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"
  2100. />
  2101. </div>
  2102. <div>
  2103. <label className="block text-sm text-bambu-gray mb-1">
  2104. Topic prefix
  2105. </label>
  2106. <input
  2107. type="text"
  2108. value={localSettings.mqtt_topic_prefix ?? 'bambuddy'}
  2109. onChange={(e) => updateSetting('mqtt_topic_prefix', e.target.value)}
  2110. placeholder="bambuddy"
  2111. 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"
  2112. />
  2113. <p className="text-xs text-bambu-gray mt-1">
  2114. Topics will be: {localSettings.mqtt_topic_prefix || 'bambuddy'}/printers/&lt;serial&gt;/status, etc.
  2115. </p>
  2116. </div>
  2117. {/* Connection Info */}
  2118. {mqttStatus && (
  2119. <div className="pt-3 mt-3 border-t border-bambu-dark-tertiary">
  2120. <div className="flex items-center gap-2 text-sm">
  2121. <span className={`w-2 h-2 rounded-full ${mqttStatus.connected ? 'bg-green-400' : 'bg-red-400'}`} />
  2122. <span className="text-bambu-gray">
  2123. {mqttStatus.connected ? (
  2124. <>{t('settings.mqttConnectedTo')} <span className="text-white">{mqttStatus.broker}:{mqttStatus.port}</span></>
  2125. ) : (
  2126. t('settings.spoolmanDisconnected')
  2127. )}
  2128. </span>
  2129. </div>
  2130. </div>
  2131. )}
  2132. </div>
  2133. )}
  2134. </CardContent>
  2135. </Card>
  2136. </div>
  2137. {/* Third Column - Prometheus Metrics */}
  2138. <div className="flex-1 lg:max-w-md space-y-4">
  2139. <Card>
  2140. <CardHeader>
  2141. <h2 className="text-lg font-semibold text-white flex items-center gap-2">
  2142. <TrendingUp className="w-5 h-5 text-orange-400" />
  2143. Prometheus Metrics
  2144. </h2>
  2145. </CardHeader>
  2146. <CardContent className="space-y-4">
  2147. <p className="text-sm text-bambu-gray">
  2148. Expose printer metrics at <code className="bg-bambu-dark px-1 rounded">/api/v1/metrics</code> for Prometheus/Grafana monitoring.
  2149. </p>
  2150. <div className="flex items-center justify-between">
  2151. <div>
  2152. <p className="text-white">{t('settings.enableMetricsEndpoint')}</p>
  2153. <p className="text-xs text-bambu-gray">{t('settings.prometheusDescription')}</p>
  2154. </div>
  2155. <label className="relative inline-flex items-center cursor-pointer">
  2156. <input
  2157. type="checkbox"
  2158. checked={localSettings.prometheus_enabled ?? false}
  2159. onChange={(e) => updateSetting('prometheus_enabled', e.target.checked)}
  2160. className="sr-only peer"
  2161. />
  2162. <div className="w-11 h-6 bg-bambu-dark-tertiary peer-focus:outline-none rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-bambu-green"></div>
  2163. </label>
  2164. </div>
  2165. {localSettings.prometheus_enabled && (
  2166. <div className="space-y-4 pt-2 border-t border-bambu-dark-tertiary">
  2167. <div>
  2168. <label className="block text-sm text-bambu-gray mb-1">
  2169. Bearer Token (optional)
  2170. </label>
  2171. <input
  2172. type="password"
  2173. value={localSettings.prometheus_token ?? ''}
  2174. onChange={(e) => updateSetting('prometheus_token', e.target.value)}
  2175. placeholder={t('settings.leaveEmptyForNoAuth')}
  2176. 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"
  2177. />
  2178. <p className="text-xs text-bambu-gray mt-1">
  2179. If set, requests must include <code className="bg-bambu-dark px-1 rounded">Authorization: Bearer &lt;token&gt;</code>
  2180. </p>
  2181. </div>
  2182. <div className="pt-2 border-t border-bambu-dark-tertiary">
  2183. <p className="text-sm text-white mb-2">{t('settings.availableMetrics')}</p>
  2184. <div className="text-xs text-bambu-gray space-y-1">
  2185. <p><code className="text-orange-400">bambuddy_printer_connected</code> - Connection status</p>
  2186. <p><code className="text-orange-400">bambuddy_printer_state</code> - Printer state (idle/printing/etc)</p>
  2187. <p><code className="text-orange-400">bambuddy_print_progress</code> - Print progress 0-100%</p>
  2188. <p><code className="text-orange-400">bambuddy_bed_temp_celsius</code> - Bed temperature</p>
  2189. <p><code className="text-orange-400">bambuddy_nozzle_temp_celsius</code> - Nozzle temperature</p>
  2190. <p><code className="text-orange-400">bambuddy_prints_total</code> - Total prints by result</p>
  2191. <p className="text-bambu-gray/70 italic">...and more (layers, fans, queue, filament usage)</p>
  2192. </div>
  2193. </div>
  2194. </div>
  2195. )}
  2196. </CardContent>
  2197. </Card>
  2198. </div>
  2199. </div>
  2200. )}
  2201. {/* Home Assistant Test Connection Modal */}
  2202. {haTestResult && (
  2203. <div className="fixed inset-0 bg-black/50 flex items-center justify-center z-50">
  2204. <div className="bg-bambu-dark-secondary rounded-lg p-6 max-w-md w-full mx-4">
  2205. <div className="flex items-center gap-3 mb-4">
  2206. {haTestResult.success ? (
  2207. <CheckCircle className="w-8 h-8 text-green-400" />
  2208. ) : (
  2209. <XCircle className="w-8 h-8 text-red-400" />
  2210. )}
  2211. <h3 className="text-lg font-medium text-white">
  2212. {haTestResult.success ? 'Connection Successful' : 'Connection Failed'}
  2213. </h3>
  2214. </div>
  2215. <p className="text-bambu-gray mb-6">
  2216. {haTestResult.success
  2217. ? haTestResult.message || 'Successfully connected to Home Assistant.'
  2218. : haTestResult.error || 'Failed to connect to Home Assistant.'}
  2219. </p>
  2220. <div className="flex justify-end">
  2221. <Button
  2222. variant="primary"
  2223. onClick={() => setHaTestResult(null)}
  2224. >
  2225. OK
  2226. </Button>
  2227. </div>
  2228. </div>
  2229. </div>
  2230. )}
  2231. {/* Smart Plugs Tab */}
  2232. {activeTab === 'plugs' && (
  2233. <div className="max-w-4xl">
  2234. <div className="flex items-start justify-between mb-6">
  2235. <div>
  2236. <h2 className="text-lg font-semibold text-white flex items-center gap-2">
  2237. <Plug className="w-5 h-5 text-bambu-green" />
  2238. Smart Plugs
  2239. </h2>
  2240. <p className="text-sm text-bambu-gray mt-1">
  2241. Connect smart plugs (Tasmota or Home Assistant) to automate power control and track energy usage for your printers.
  2242. </p>
  2243. </div>
  2244. <div className="flex items-center gap-2 pt-1 shrink-0">
  2245. {smartPlugs && smartPlugs.filter(p => p.enabled).length > 1 && (
  2246. <>
  2247. <Button
  2248. variant="secondary"
  2249. size="sm"
  2250. className="whitespace-nowrap"
  2251. onClick={() => setShowBulkPlugConfirm('on')}
  2252. disabled={bulkPlugActionMutation.isPending}
  2253. title={t('settings.turnAllPlugsOn')}
  2254. >
  2255. {bulkPlugActionMutation.isPending ? (
  2256. <Loader2 className="w-4 h-4 animate-spin" />
  2257. ) : (
  2258. <Power className="w-4 h-4 text-bambu-green" />
  2259. )}
  2260. All On
  2261. </Button>
  2262. <Button
  2263. variant="secondary"
  2264. size="sm"
  2265. className="whitespace-nowrap"
  2266. onClick={() => setShowBulkPlugConfirm('off')}
  2267. disabled={bulkPlugActionMutation.isPending}
  2268. title={t('settings.turnAllPlugsOff')}
  2269. >
  2270. {bulkPlugActionMutation.isPending ? (
  2271. <Loader2 className="w-4 h-4 animate-spin" />
  2272. ) : (
  2273. <PowerOff className="w-4 h-4 text-red-400" />
  2274. )}
  2275. All Off
  2276. </Button>
  2277. </>
  2278. )}
  2279. <Button
  2280. className="whitespace-nowrap"
  2281. onClick={() => {
  2282. setEditingPlug(null);
  2283. setShowPlugModal(true);
  2284. }}
  2285. >
  2286. <Plus className="w-4 h-4" />
  2287. Add Smart Plug
  2288. </Button>
  2289. </div>
  2290. </div>
  2291. {/* Energy Summary Card */}
  2292. {smartPlugs && smartPlugs.length > 0 && (
  2293. <Card className="mb-6">
  2294. <CardHeader>
  2295. <h3 className="text-base font-semibold text-white flex items-center gap-2">
  2296. <Zap className="w-4 h-4 text-yellow-400" />
  2297. Energy Summary
  2298. {energyLoading && (
  2299. <Loader2 className="w-4 h-4 animate-spin text-bambu-gray ml-2" />
  2300. )}
  2301. </h3>
  2302. </CardHeader>
  2303. <CardContent>
  2304. {plugEnergySummary ? (
  2305. <div className="grid grid-cols-2 md:grid-cols-4 gap-4">
  2306. {/* Current Power */}
  2307. <div className="bg-bambu-dark rounded-lg p-3">
  2308. <div className="flex items-center gap-2 text-bambu-gray text-xs mb-1">
  2309. <Zap className="w-3 h-3" />
  2310. Current Power
  2311. </div>
  2312. <div className="text-xl font-bold text-white">
  2313. {plugEnergySummary.totalPower.toFixed(1)}
  2314. <span className="text-sm font-normal text-bambu-gray ml-1">W</span>
  2315. </div>
  2316. <div className="text-xs text-bambu-gray mt-1">
  2317. {plugEnergySummary.reachableCount}/{plugEnergySummary.totalPlugs} plugs online
  2318. </div>
  2319. </div>
  2320. {/* Today */}
  2321. <div className="bg-bambu-dark rounded-lg p-3">
  2322. <div className="flex items-center gap-2 text-bambu-gray text-xs mb-1">
  2323. <Calendar className="w-3 h-3" />
  2324. Today
  2325. </div>
  2326. <div className="text-xl font-bold text-white">
  2327. {plugEnergySummary.totalToday.toFixed(2)}
  2328. <span className="text-sm font-normal text-bambu-gray ml-1">kWh</span>
  2329. </div>
  2330. {(localSettings?.energy_cost_per_kwh ?? 0) > 0 && (
  2331. <div className="text-xs text-bambu-gray mt-1">
  2332. ~{(plugEnergySummary.totalToday * (localSettings?.energy_cost_per_kwh ?? 0)).toFixed(2)} {localSettings?.currency}
  2333. </div>
  2334. )}
  2335. </div>
  2336. {/* Yesterday */}
  2337. <div className="bg-bambu-dark rounded-lg p-3">
  2338. <div className="flex items-center gap-2 text-bambu-gray text-xs mb-1">
  2339. <TrendingUp className="w-3 h-3" />
  2340. Yesterday
  2341. </div>
  2342. <div className="text-xl font-bold text-white">
  2343. {plugEnergySummary.totalYesterday.toFixed(2)}
  2344. <span className="text-sm font-normal text-bambu-gray ml-1">kWh</span>
  2345. </div>
  2346. {(localSettings?.energy_cost_per_kwh ?? 0) > 0 && (
  2347. <div className="text-xs text-bambu-gray mt-1">
  2348. ~{(plugEnergySummary.totalYesterday * (localSettings?.energy_cost_per_kwh ?? 0)).toFixed(2)} {localSettings?.currency}
  2349. </div>
  2350. )}
  2351. </div>
  2352. {/* Total Lifetime */}
  2353. <div className="bg-bambu-dark rounded-lg p-3">
  2354. <div className="flex items-center gap-2 text-bambu-gray text-xs mb-1">
  2355. <DollarSign className="w-3 h-3" />
  2356. Total
  2357. </div>
  2358. <div className="text-xl font-bold text-white">
  2359. {plugEnergySummary.totalLifetime.toFixed(1)}
  2360. <span className="text-sm font-normal text-bambu-gray ml-1">kWh</span>
  2361. </div>
  2362. {(localSettings?.energy_cost_per_kwh ?? 0) > 0 && (
  2363. <div className="text-xs text-bambu-gray mt-1">
  2364. ~{(plugEnergySummary.totalLifetime * (localSettings?.energy_cost_per_kwh ?? 0)).toFixed(2)} {localSettings?.currency}
  2365. </div>
  2366. )}
  2367. </div>
  2368. </div>
  2369. ) : !energyLoading ? (
  2370. <p className="text-sm text-bambu-gray">
  2371. Enable plugs to see energy summary
  2372. </p>
  2373. ) : null}
  2374. </CardContent>
  2375. </Card>
  2376. )}
  2377. {plugsLoading ? (
  2378. <div className="flex justify-center py-12">
  2379. <Loader2 className="w-8 h-8 text-bambu-green animate-spin" />
  2380. </div>
  2381. ) : smartPlugs && smartPlugs.length > 0 ? (
  2382. <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
  2383. {smartPlugs.map((plug) => (
  2384. <SmartPlugCard
  2385. key={plug.id}
  2386. plug={plug}
  2387. onEdit={(p) => {
  2388. setEditingPlug(p);
  2389. setShowPlugModal(true);
  2390. }}
  2391. />
  2392. ))}
  2393. </div>
  2394. ) : (
  2395. <Card>
  2396. <CardContent className="py-12">
  2397. <div className="text-center text-bambu-gray">
  2398. <Plug className="w-16 h-16 mx-auto mb-4 opacity-30" />
  2399. <p className="text-lg font-medium text-white mb-2">{t('settings.noSmartPlugsTitle')}</p>
  2400. <p className="text-sm mb-4">{t('settings.noSmartPlugsDescription')}</p>
  2401. <Button
  2402. onClick={() => {
  2403. setEditingPlug(null);
  2404. setShowPlugModal(true);
  2405. }}
  2406. >
  2407. <Plus className="w-4 h-4" />
  2408. {t('settings.addFirstSmartPlug')}
  2409. </Button>
  2410. </div>
  2411. </CardContent>
  2412. </Card>
  2413. )}
  2414. </div>
  2415. )}
  2416. {/* Notifications Tab */}
  2417. {activeTab === 'notifications' && (
  2418. <div className="grid grid-cols-1 lg:grid-cols-2 gap-8">
  2419. {/* Left Column: Providers */}
  2420. <div>
  2421. <div className="flex items-center justify-between mb-4">
  2422. <h2 className="text-lg font-semibold text-white flex items-center gap-2">
  2423. <Bell className="w-5 h-5 text-bambu-green" />
  2424. {t('settings.providers')}
  2425. </h2>
  2426. <div className="flex items-center gap-2">
  2427. <Button
  2428. size="sm"
  2429. variant="secondary"
  2430. onClick={() => setShowLogViewer(true)}
  2431. >
  2432. <History className="w-4 h-4" />
  2433. {t('settings.log')}
  2434. </Button>
  2435. {notificationProviders && notificationProviders.length > 0 && (
  2436. <Button
  2437. size="sm"
  2438. variant="secondary"
  2439. onClick={() => {
  2440. setTestAllResult(null);
  2441. testAllMutation.mutate();
  2442. }}
  2443. disabled={testAllMutation.isPending}
  2444. >
  2445. {testAllMutation.isPending ? (
  2446. <Loader2 className="w-4 h-4 animate-spin" />
  2447. ) : (
  2448. <Send className="w-4 h-4" />
  2449. )}
  2450. {t('settings.testAll')}
  2451. </Button>
  2452. )}
  2453. <Button
  2454. size="sm"
  2455. onClick={() => {
  2456. setEditingProvider(null);
  2457. setShowNotificationModal(true);
  2458. }}
  2459. >
  2460. <Plus className="w-4 h-4" />
  2461. Add
  2462. </Button>
  2463. </div>
  2464. </div>
  2465. {/* Notification Language Setting */}
  2466. <Card className="mb-4">
  2467. <CardContent className="py-3">
  2468. <div className="flex items-center justify-between">
  2469. <div>
  2470. <p className="text-white text-sm font-medium">{t('settings.notificationLanguage')}</p>
  2471. <p className="text-xs text-bambu-gray">{t('settings.notificationLanguageDescription')}</p>
  2472. </div>
  2473. <select
  2474. value={localSettings.notification_language || 'en'}
  2475. onChange={(e) => updateSetting('notification_language', e.target.value)}
  2476. className="px-2 py-1.5 bg-bambu-dark border border-bambu-dark-tertiary rounded text-white text-sm focus:outline-none focus:ring-1 focus:ring-bambu-green"
  2477. >
  2478. {availableLanguages.map((lang) => (
  2479. <option key={lang.code} value={lang.code}>
  2480. {lang.nativeName}
  2481. </option>
  2482. ))}
  2483. </select>
  2484. </div>
  2485. </CardContent>
  2486. </Card>
  2487. {/* Test All Results */}
  2488. {testAllResult && (
  2489. <Card className="mb-4">
  2490. <CardContent className="py-3">
  2491. <div className="flex items-center justify-between mb-2">
  2492. <span className="text-sm font-medium text-white">{t('settings.testResults')}</span>
  2493. <button
  2494. onClick={() => setTestAllResult(null)}
  2495. className="text-bambu-gray hover:text-white text-xs"
  2496. >
  2497. {t('common.dismiss')}
  2498. </button>
  2499. </div>
  2500. <div className="flex items-center gap-4 text-sm mb-2">
  2501. <span className="flex items-center gap-1 text-bambu-green">
  2502. <CheckCircle className="w-4 h-4" />
  2503. {t('settings.testPassedCount', { count: testAllResult.success })}
  2504. </span>
  2505. {testAllResult.failed > 0 && (
  2506. <span className="flex items-center gap-1 text-red-400">
  2507. <XCircle className="w-4 h-4" />
  2508. {t('settings.testFailedCount', { count: testAllResult.failed })}
  2509. </span>
  2510. )}
  2511. </div>
  2512. {testAllResult.results.filter(r => !r.success).length > 0 && (
  2513. <div className="space-y-1 mt-2 pt-2 border-t border-bambu-dark-tertiary">
  2514. {testAllResult.results.filter(r => !r.success).map((result) => (
  2515. <div key={result.provider_id} className="text-xs text-red-400">
  2516. <span className="font-medium">{result.provider_name}:</span> {result.message}
  2517. </div>
  2518. ))}
  2519. </div>
  2520. )}
  2521. </CardContent>
  2522. </Card>
  2523. )}
  2524. {providersLoading ? (
  2525. <div className="flex justify-center py-12">
  2526. <Loader2 className="w-6 h-6 text-bambu-green animate-spin" />
  2527. </div>
  2528. ) : notificationProviders && notificationProviders.length > 0 ? (
  2529. <div className="space-y-3">
  2530. {notificationProviders.map((provider) => (
  2531. <NotificationProviderCard
  2532. key={provider.id}
  2533. provider={provider}
  2534. onEdit={(p) => {
  2535. setEditingProvider(p);
  2536. setShowNotificationModal(true);
  2537. }}
  2538. />
  2539. ))}
  2540. </div>
  2541. ) : (
  2542. <Card>
  2543. <CardContent className="py-8">
  2544. <div className="text-center text-bambu-gray">
  2545. <Bell className="w-12 h-12 mx-auto mb-3 opacity-30" />
  2546. <p className="text-sm font-medium text-white mb-2">{t('settings.noProvidersTitle')}</p>
  2547. <p className="text-xs mb-3">{t('settings.noProvidersDescription')}</p>
  2548. <Button
  2549. size="sm"
  2550. onClick={() => {
  2551. setEditingProvider(null);
  2552. setShowNotificationModal(true);
  2553. }}
  2554. >
  2555. <Plus className="w-4 h-4" />
  2556. {t('settings.addProvider')}
  2557. </Button>
  2558. </div>
  2559. </CardContent>
  2560. </Card>
  2561. )}
  2562. </div>
  2563. {/* Right Column: Templates */}
  2564. <div>
  2565. <h2 className="text-lg font-semibold text-white flex items-center gap-2 mb-4">
  2566. <FileText className="w-5 h-5 text-bambu-green" />
  2567. {t('settings.messageTemplates')}
  2568. </h2>
  2569. <p className="text-sm text-bambu-gray mb-4">
  2570. {t('settings.messageTemplatesDescription')}
  2571. </p>
  2572. {templatesLoading ? (
  2573. <div className="flex justify-center py-8">
  2574. <Loader2 className="w-6 h-6 text-bambu-green animate-spin" />
  2575. </div>
  2576. ) : notificationTemplates && notificationTemplates.length > 0 ? (
  2577. <div className="space-y-2">
  2578. {notificationTemplates.map((template) => (
  2579. <Card
  2580. key={template.id}
  2581. className="cursor-pointer hover:border-bambu-green/50 transition-colors"
  2582. onClick={() => setEditingTemplate(template)}
  2583. >
  2584. <CardContent className="py-2.5 px-3">
  2585. <div className="flex items-center justify-between">
  2586. <div className="min-w-0 flex-1">
  2587. <p className="text-white font-medium text-sm truncate">{template.name}</p>
  2588. <p className="text-bambu-gray text-xs truncate mt-0.5">
  2589. {template.title_template}
  2590. </p>
  2591. </div>
  2592. <button
  2593. className="p-1.5 hover:bg-bambu-dark-tertiary rounded transition-colors shrink-0 ml-2"
  2594. onClick={(e) => {
  2595. e.stopPropagation();
  2596. setEditingTemplate(template);
  2597. }}
  2598. >
  2599. <Edit2 className="w-4 h-4 text-bambu-gray" />
  2600. </button>
  2601. </div>
  2602. </CardContent>
  2603. </Card>
  2604. ))}
  2605. </div>
  2606. ) : (
  2607. <Card>
  2608. <CardContent className="py-8">
  2609. <div className="text-center text-bambu-gray">
  2610. <FileText className="w-12 h-12 mx-auto mb-3 opacity-30" />
  2611. <p className="text-sm">{t('settings.noTemplatesAvailable')}</p>
  2612. </div>
  2613. </CardContent>
  2614. </Card>
  2615. )}
  2616. </div>
  2617. </div>
  2618. )}
  2619. {/* API Keys Tab */}
  2620. {activeTab === 'apikeys' && (
  2621. <div className="grid grid-cols-1 xl:grid-cols-2 gap-8">
  2622. {/* Left Column - API Keys Management */}
  2623. <div>
  2624. <div className="flex items-start justify-between gap-4 mb-6">
  2625. <div className="flex-1">
  2626. <h2 className="text-lg font-semibold text-white flex items-center gap-2">
  2627. <Key className="w-5 h-5 text-bambu-green" />
  2628. {t('settings.apiKeys')}
  2629. </h2>
  2630. <p className="text-sm text-bambu-gray mt-1">
  2631. {t('settings.apiKeysDescription')}
  2632. </p>
  2633. </div>
  2634. <Button size="sm" onClick={() => setShowCreateAPIKey(true)} className="flex-shrink-0">
  2635. <Plus className="w-4 h-4" />
  2636. {t('settings.createKey')}
  2637. </Button>
  2638. </div>
  2639. {/* Created Key Display */}
  2640. {createdAPIKey && (
  2641. <Card className="mb-6 border-bambu-green">
  2642. <CardContent className="py-4">
  2643. <div className="flex items-start gap-3">
  2644. <CheckCircle className="w-5 h-5 text-bambu-green flex-shrink-0 mt-0.5" />
  2645. <div className="flex-1">
  2646. <p className="text-white font-medium mb-1">{t('settings.apiKeyCreated')}</p>
  2647. <p className="text-sm text-bambu-gray mb-2">
  2648. {t('settings.apiKeyCopyWarning')}
  2649. </p>
  2650. <div className="flex items-center gap-2 bg-bambu-dark rounded-lg p-2">
  2651. <code className="flex-1 text-sm text-bambu-green font-mono break-all">
  2652. {createdAPIKey}
  2653. </code>
  2654. <Button
  2655. variant="secondary"
  2656. size="sm"
  2657. onClick={async () => {
  2658. try {
  2659. if (navigator.clipboard && navigator.clipboard.writeText) {
  2660. await navigator.clipboard.writeText(createdAPIKey);
  2661. } else {
  2662. const textArea = document.createElement('textarea');
  2663. textArea.value = createdAPIKey;
  2664. textArea.style.position = 'fixed';
  2665. textArea.style.left = '-999999px';
  2666. document.body.appendChild(textArea);
  2667. textArea.select();
  2668. document.execCommand('copy');
  2669. document.body.removeChild(textArea);
  2670. }
  2671. showToast(t('settings.toast.keyCopied'));
  2672. } catch {
  2673. showToast(t('settings.toast.copyFailed'), 'error');
  2674. }
  2675. }}
  2676. >
  2677. <Copy className="w-4 h-4" />
  2678. </Button>
  2679. </div>
  2680. <div className="flex gap-2 mt-3">
  2681. <Button
  2682. variant="secondary"
  2683. size="sm"
  2684. onClick={() => {
  2685. setTestApiKey(createdAPIKey);
  2686. showToast(t('settings.toast.keyAddedToBrowser'));
  2687. }}
  2688. >
  2689. {t('settings.useInApiBrowser')}
  2690. </Button>
  2691. <Button
  2692. variant="secondary"
  2693. size="sm"
  2694. onClick={() => setCreatedAPIKey(null)}
  2695. >
  2696. {t('common.dismiss')}
  2697. </Button>
  2698. </div>
  2699. </div>
  2700. </div>
  2701. </CardContent>
  2702. </Card>
  2703. )}
  2704. {/* Create Key Form */}
  2705. {showCreateAPIKey && (
  2706. <Card className="mb-6">
  2707. <CardHeader>
  2708. <h3 className="text-base font-semibold text-white">{t('settings.createNewApiKey')}</h3>
  2709. </CardHeader>
  2710. <CardContent className="space-y-4">
  2711. <div>
  2712. <label className="block text-sm text-bambu-gray mb-1">{t('settings.keyName')}</label>
  2713. <input
  2714. type="text"
  2715. value={newAPIKeyName}
  2716. onChange={(e) => setNewAPIKeyName(e.target.value)}
  2717. placeholder={t('settings.keyNamePlaceholder')}
  2718. 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"
  2719. />
  2720. </div>
  2721. <div>
  2722. <label className="block text-sm text-bambu-gray mb-2">{t('common.permissions')}</label>
  2723. <div className="space-y-2">
  2724. <label className="flex items-center gap-3 cursor-pointer">
  2725. <input
  2726. type="checkbox"
  2727. checked={newAPIKeyPermissions.can_read_status}
  2728. onChange={(e) => setNewAPIKeyPermissions(prev => ({ ...prev, can_read_status: e.target.checked }))}
  2729. className="w-4 h-4 text-bambu-green rounded border-bambu-dark-tertiary bg-bambu-dark focus:ring-bambu-green"
  2730. />
  2731. <div>
  2732. <span className="text-white">{t('settings.readStatus')}</span>
  2733. <p className="text-xs text-bambu-gray">{t('settings.readStatusDescription')}</p>
  2734. </div>
  2735. </label>
  2736. <label className="flex items-center gap-3 cursor-pointer">
  2737. <input
  2738. type="checkbox"
  2739. checked={newAPIKeyPermissions.can_queue}
  2740. onChange={(e) => setNewAPIKeyPermissions(prev => ({ ...prev, can_queue: e.target.checked }))}
  2741. className="w-4 h-4 text-bambu-green rounded border-bambu-dark-tertiary bg-bambu-dark focus:ring-bambu-green"
  2742. />
  2743. <div>
  2744. <span className="text-white">{t('settings.manageQueue')}</span>
  2745. <p className="text-xs text-bambu-gray">{t('settings.manageQueueDescription')}</p>
  2746. </div>
  2747. </label>
  2748. <label className="flex items-center gap-3 cursor-pointer">
  2749. <input
  2750. type="checkbox"
  2751. checked={newAPIKeyPermissions.can_control_printer}
  2752. onChange={(e) => setNewAPIKeyPermissions(prev => ({ ...prev, can_control_printer: e.target.checked }))}
  2753. className="w-4 h-4 text-bambu-green rounded border-bambu-dark-tertiary bg-bambu-dark focus:ring-bambu-green"
  2754. />
  2755. <div>
  2756. <span className="text-white">{t('settings.controlPrinter')}</span>
  2757. <p className="text-xs text-bambu-gray">{t('settings.controlPrinterDescription')}</p>
  2758. </div>
  2759. </label>
  2760. </div>
  2761. </div>
  2762. <div className="flex items-center gap-2 pt-2">
  2763. <Button
  2764. onClick={() => createAPIKeyMutation.mutate({
  2765. name: newAPIKeyName || t('settings.unnamedKey'),
  2766. ...newAPIKeyPermissions,
  2767. })}
  2768. disabled={createAPIKeyMutation.isPending}
  2769. >
  2770. {createAPIKeyMutation.isPending ? (
  2771. <Loader2 className="w-4 h-4 animate-spin" />
  2772. ) : (
  2773. <Plus className="w-4 h-4" />
  2774. )}
  2775. {t('settings.createKey')}
  2776. </Button>
  2777. <Button variant="secondary" onClick={() => setShowCreateAPIKey(false)}>
  2778. {t('common.cancel')}
  2779. </Button>
  2780. </div>
  2781. </CardContent>
  2782. </Card>
  2783. )}
  2784. {/* Existing Keys List */}
  2785. {apiKeysLoading ? (
  2786. <div className="flex justify-center py-12">
  2787. <Loader2 className="w-8 h-8 text-bambu-green animate-spin" />
  2788. </div>
  2789. ) : apiKeys && apiKeys.length > 0 ? (
  2790. <div className="space-y-3">
  2791. {apiKeys.map((key) => (
  2792. <Card key={key.id}>
  2793. <CardContent className="py-3">
  2794. <div className="flex items-center justify-between">
  2795. <div className="flex items-center gap-3">
  2796. <Key className={`w-5 h-5 ${key.enabled ? 'text-bambu-green' : 'text-bambu-gray'}`} />
  2797. <div>
  2798. <p className="text-white font-medium">{key.name}</p>
  2799. <p className="text-xs text-bambu-gray">
  2800. {key.key_prefix}••••••••
  2801. {key.last_used && ` · ${t('settings.lastUsed')}: ${formatDateOnly(key.last_used)}`}
  2802. </p>
  2803. </div>
  2804. </div>
  2805. <div className="flex items-center gap-2">
  2806. <div className="flex gap-1 text-xs">
  2807. {key.can_read_status && (
  2808. <span className="px-1.5 py-0.5 bg-blue-500/20 text-blue-400 rounded">{t('settings.read')}</span>
  2809. )}
  2810. {key.can_queue && (
  2811. <span className="px-1.5 py-0.5 bg-green-500/20 text-green-400 rounded">{t('queue.title')}</span>
  2812. )}
  2813. {key.can_control_printer && (
  2814. <span className="px-1.5 py-0.5 bg-orange-500/20 text-orange-400 rounded">{t('settings.control')}</span>
  2815. )}
  2816. </div>
  2817. <Button
  2818. variant="secondary"
  2819. size="sm"
  2820. onClick={() => setShowDeleteAPIKeyConfirm(key.id)}
  2821. >
  2822. <Trash2 className="w-4 h-4 text-red-400" />
  2823. </Button>
  2824. </div>
  2825. </div>
  2826. </CardContent>
  2827. </Card>
  2828. ))}
  2829. </div>
  2830. ) : (
  2831. <Card>
  2832. <CardContent className="py-12">
  2833. <div className="text-center text-bambu-gray">
  2834. <Key className="w-16 h-16 mx-auto mb-4 opacity-30" />
  2835. <p className="text-lg font-medium text-white mb-2">{t('settings.apiKeysEmptyTitle')}</p>
  2836. <p className="text-sm mb-4">{t('settings.apiKeysEmptyDescription')}</p>
  2837. <Button onClick={() => setShowCreateAPIKey(true)}>
  2838. <Plus className="w-4 h-4" />
  2839. {t('settings.createFirstKey')}
  2840. </Button>
  2841. </div>
  2842. </CardContent>
  2843. </Card>
  2844. )}
  2845. {/* Webhook Documentation */}
  2846. <Card className="mt-6">
  2847. <CardHeader>
  2848. <h3 className="text-base font-semibold text-white">{t('settings.webhookEndpoints')}</h3>
  2849. </CardHeader>
  2850. <CardContent className="space-y-3 text-sm">
  2851. <p className="text-bambu-gray">
  2852. {t('settings.webhookApiKeyHint')}
  2853. </p>
  2854. <div className="space-y-2 font-mono text-xs">
  2855. <div className="p-2 bg-bambu-dark rounded">
  2856. <span className="text-blue-400">GET</span>{' '}
  2857. <span className="text-white">/api/v1/webhook/status</span>
  2858. <span className="text-bambu-gray"> - {t('settings.webhook.getAllStatus')}</span>
  2859. </div>
  2860. <div className="p-2 bg-bambu-dark rounded">
  2861. <span className="text-blue-400">GET</span>{' '}
  2862. <span className="text-white">/api/v1/webhook/status/:id</span>
  2863. <span className="text-bambu-gray"> - {t('settings.webhook.getSpecificStatus')}</span>
  2864. </div>
  2865. <div className="p-2 bg-bambu-dark rounded">
  2866. <span className="text-green-400">POST</span>{' '}
  2867. <span className="text-white">/api/v1/webhook/queue</span>
  2868. <span className="text-bambu-gray"> - {t('settings.webhook.addToQueue')}</span>
  2869. </div>
  2870. <div className="p-2 bg-bambu-dark rounded">
  2871. <span className="text-orange-400">POST</span>{' '}
  2872. <span className="text-white">/api/v1/webhook/printer/:id/pause</span>
  2873. <span className="text-bambu-gray"> - {t('settings.webhook.pausePrint')}</span>
  2874. </div>
  2875. <div className="p-2 bg-bambu-dark rounded">
  2876. <span className="text-orange-400">POST</span>{' '}
  2877. <span className="text-white">/api/v1/webhook/printer/:id/resume</span>
  2878. <span className="text-bambu-gray"> - {t('settings.webhook.resumePrint')}</span>
  2879. </div>
  2880. <div className="p-2 bg-bambu-dark rounded">
  2881. <span className="text-red-400">POST</span>{' '}
  2882. <span className="text-white">/api/v1/webhook/printer/:id/stop</span>
  2883. <span className="text-bambu-gray"> - {t('settings.webhook.stopPrint')}</span>
  2884. </div>
  2885. </div>
  2886. </CardContent>
  2887. </Card>
  2888. </div>
  2889. {/* Right Column - API Browser */}
  2890. <div>
  2891. <div className="mb-6">
  2892. <h2 className="text-lg font-semibold text-white flex items-center gap-2">
  2893. <Globe className="w-5 h-5 text-bambu-green" />
  2894. {t('settings.apiBrowser')}
  2895. </h2>
  2896. <p className="text-sm text-bambu-gray mt-1">
  2897. {t('settings.apiBrowserDescription')}
  2898. </p>
  2899. </div>
  2900. {/* API Key Input for Testing */}
  2901. <Card className="mb-4">
  2902. <CardContent className="py-3">
  2903. <label className="block text-sm text-bambu-gray mb-2">{t('settings.apiKeyForTesting')}</label>
  2904. <input
  2905. type="text"
  2906. value={testApiKey}
  2907. onChange={(e) => setTestApiKey(e.target.value)}
  2908. placeholder={t('settings.apiKeyPlaceholder')}
  2909. className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white font-mono text-sm focus:border-bambu-green focus:outline-none"
  2910. />
  2911. <p className="text-xs text-bambu-gray mt-2">
  2912. {t('settings.apiKeyHint')}
  2913. </p>
  2914. </CardContent>
  2915. </Card>
  2916. <APIBrowser apiKey={testApiKey} />
  2917. </div>
  2918. </div>
  2919. )}
  2920. {/* Virtual Printer Tab */}
  2921. {activeTab === 'virtual-printer' && (
  2922. <VirtualPrinterSettings />
  2923. )}
  2924. {/* Filament Tab */}
  2925. {activeTab === 'filament' && localSettings && (
  2926. <div className="flex flex-col lg:flex-row gap-6 lg:gap-8">
  2927. {/* Left Column - AMS Display Thresholds */}
  2928. <div className="flex-1 lg:max-w-xl">
  2929. <Card>
  2930. <CardHeader>
  2931. <h2 className="text-lg font-semibold text-white">{t('settings.amsDisplayThresholds')}</h2>
  2932. </CardHeader>
  2933. <CardContent className="space-y-4">
  2934. <p className="text-sm text-bambu-gray">
  2935. {t('settings.amsThresholdsDescription')}
  2936. </p>
  2937. {/* Humidity Thresholds */}
  2938. <div className="space-y-3">
  2939. <div className="flex items-center gap-2 text-white">
  2940. <Droplets className="w-4 h-4 text-blue-400" />
  2941. <span className="font-medium">{t('settings.humidity')}</span>
  2942. </div>
  2943. <div className="grid grid-cols-2 gap-3">
  2944. <div>
  2945. <label className="block text-sm text-bambu-gray mb-1">
  2946. {t('settings.goodGreen')} ≤
  2947. </label>
  2948. <div className="flex items-center gap-2">
  2949. <input
  2950. type="number"
  2951. min="0"
  2952. max="100"
  2953. value={localSettings.ams_humidity_good ?? 40}
  2954. onChange={(e) => updateSetting('ams_humidity_good', parseInt(e.target.value) || 40)}
  2955. 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"
  2956. />
  2957. <span className="text-bambu-gray">%</span>
  2958. </div>
  2959. </div>
  2960. <div>
  2961. <label className="block text-sm text-bambu-gray mb-1">
  2962. {t('settings.fairOrange')} ≤
  2963. </label>
  2964. <div className="flex items-center gap-2">
  2965. <input
  2966. type="number"
  2967. min="0"
  2968. max="100"
  2969. value={localSettings.ams_humidity_fair ?? 60}
  2970. onChange={(e) => updateSetting('ams_humidity_fair', parseInt(e.target.value) || 60)}
  2971. 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"
  2972. />
  2973. <span className="text-bambu-gray">%</span>
  2974. </div>
  2975. </div>
  2976. </div>
  2977. <p className="text-xs text-bambu-gray">
  2978. {t('settings.aboveFairBad')}
  2979. </p>
  2980. </div>
  2981. {/* Temperature Thresholds */}
  2982. <div className="space-y-3 pt-2 border-t border-bambu-dark-tertiary">
  2983. <div className="flex items-center gap-2 text-white">
  2984. <Thermometer className="w-4 h-4 text-orange-400" />
  2985. <span className="font-medium">{t('settings.temperature')}</span>
  2986. </div>
  2987. <div className="grid grid-cols-2 gap-3">
  2988. <div>
  2989. <label className="block text-sm text-bambu-gray mb-1">
  2990. {t('settings.goodBlue')} ≤
  2991. </label>
  2992. <div className="flex items-center gap-2">
  2993. <input
  2994. type="number"
  2995. step="0.5"
  2996. min="0"
  2997. max="60"
  2998. value={localSettings.ams_temp_good ?? 28}
  2999. onChange={(e) => updateSetting('ams_temp_good', parseFloat(e.target.value) || 28)}
  3000. 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"
  3001. />
  3002. <span className="text-bambu-gray">°C</span>
  3003. </div>
  3004. </div>
  3005. <div>
  3006. <label className="block text-sm text-bambu-gray mb-1">
  3007. {t('settings.fairOrange')} ≤
  3008. </label>
  3009. <div className="flex items-center gap-2">
  3010. <input
  3011. type="number"
  3012. step="0.5"
  3013. min="0"
  3014. max="60"
  3015. value={localSettings.ams_temp_fair ?? 35}
  3016. onChange={(e) => updateSetting('ams_temp_fair', parseFloat(e.target.value) || 35)}
  3017. 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"
  3018. />
  3019. <span className="text-bambu-gray">°C</span>
  3020. </div>
  3021. </div>
  3022. </div>
  3023. <p className="text-xs text-bambu-gray">
  3024. {t('settings.aboveFairHot')}
  3025. </p>
  3026. </div>
  3027. {/* History Retention */}
  3028. <div className="space-y-3 pt-4 border-t border-bambu-dark-tertiary">
  3029. <div className="flex items-center gap-2 text-white">
  3030. <Database className="w-4 h-4 text-purple-400" />
  3031. <span className="font-medium">{t('settings.historyRetention')}</span>
  3032. </div>
  3033. <div>
  3034. <label className="block text-sm text-bambu-gray mb-1">
  3035. {t('settings.keepSensorHistory')}
  3036. </label>
  3037. <div className="flex items-center gap-2">
  3038. <input
  3039. type="number"
  3040. min="1"
  3041. max="365"
  3042. value={localSettings.ams_history_retention_days ?? 30}
  3043. onChange={(e) => updateSetting('ams_history_retention_days', parseInt(e.target.value) || 30)}
  3044. className="w-24 px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  3045. />
  3046. <span className="text-bambu-gray">{t('common.days')}</span>
  3047. </div>
  3048. </div>
  3049. <p className="text-xs text-bambu-gray">
  3050. {t('settings.historyRetentionDescription')}
  3051. </p>
  3052. </div>
  3053. {/* Per-Printer Mapping Default */}
  3054. <div className="space-y-3 pt-4 border-t border-bambu-dark-tertiary">
  3055. <div className="flex items-center gap-2 text-white">
  3056. <Printer className="w-4 h-4 text-bambu-green" />
  3057. <span className="font-medium">{t('settings.printModal')}</span>
  3058. </div>
  3059. <div className="flex items-center justify-between">
  3060. <div>
  3061. <label className="block text-sm text-white">
  3062. {t('settings.expandCustomMapping')}
  3063. </label>
  3064. <p className="text-xs text-bambu-gray mt-0.5">
  3065. {t('settings.expandCustomMappingDescription')}
  3066. </p>
  3067. </div>
  3068. <label className="relative inline-flex items-center cursor-pointer">
  3069. <input
  3070. type="checkbox"
  3071. checked={localSettings.per_printer_mapping_expanded ?? false}
  3072. onChange={(e) => updateSetting('per_printer_mapping_expanded', e.target.checked)}
  3073. className="sr-only peer"
  3074. />
  3075. <div className="w-11 h-6 bg-bambu-dark-tertiary peer-focus:outline-none rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-bambu-green"></div>
  3076. </label>
  3077. </div>
  3078. </div>
  3079. </CardContent>
  3080. </Card>
  3081. </div>
  3082. {/* Right Column - Spoolman Integration */}
  3083. <div className="flex-1 lg:max-w-xl">
  3084. <SpoolmanSettings />
  3085. </div>
  3086. </div>
  3087. )}
  3088. {/* Delete API Key Confirmation */}
  3089. {showDeleteAPIKeyConfirm !== null && (
  3090. <ConfirmModal
  3091. title={t('settings.deleteApiKeyTitle')}
  3092. message={t('settings.deleteApiKeyMessage')}
  3093. confirmText={t('settings.deleteKey')}
  3094. variant="danger"
  3095. onConfirm={() => {
  3096. deleteAPIKeyMutation.mutate(showDeleteAPIKeyConfirm);
  3097. setShowDeleteAPIKeyConfirm(null);
  3098. }}
  3099. onCancel={() => setShowDeleteAPIKeyConfirm(null)}
  3100. />
  3101. )}
  3102. {/* Smart Plug Modal */}
  3103. {showPlugModal && (
  3104. <AddSmartPlugModal
  3105. plug={editingPlug}
  3106. onClose={() => {
  3107. setShowPlugModal(false);
  3108. setEditingPlug(null);
  3109. }}
  3110. />
  3111. )}
  3112. {/* Notification Modal */}
  3113. {showNotificationModal && (
  3114. <AddNotificationModal
  3115. provider={editingProvider}
  3116. onClose={() => {
  3117. setShowNotificationModal(false);
  3118. setEditingProvider(null);
  3119. }}
  3120. />
  3121. )}
  3122. {/* Template Editor Modal */}
  3123. {editingTemplate && (
  3124. <NotificationTemplateEditor
  3125. template={editingTemplate}
  3126. onClose={() => setEditingTemplate(null)}
  3127. />
  3128. )}
  3129. {/* Notification Log Viewer */}
  3130. {showLogViewer && (
  3131. <NotificationLogViewer
  3132. onClose={() => setShowLogViewer(false)}
  3133. />
  3134. )}
  3135. {/* Confirm Modal: Clear Notification Logs */}
  3136. {showClearLogsConfirm && (
  3137. <ConfirmModal
  3138. title={t('settings.clearNotificationLogs')}
  3139. message={t('settings.clearLogsMessage')}
  3140. confirmText={t('settings.clearLogs')}
  3141. variant="warning"
  3142. onConfirm={async () => {
  3143. setShowClearLogsConfirm(false);
  3144. try {
  3145. const result = await api.clearNotificationLogs(30);
  3146. showToast(result.message, 'success');
  3147. } catch {
  3148. showToast(t('settings.toast.clearLogsFailed'), 'error');
  3149. }
  3150. }}
  3151. onCancel={() => setShowClearLogsConfirm(false)}
  3152. />
  3153. )}
  3154. {/* Confirm Modal: Clear Local Storage */}
  3155. {showClearStorageConfirm && (
  3156. <ConfirmModal
  3157. title={t('settings.resetUiPreferences')}
  3158. message={t('settings.resetUiPreferencesMessage')}
  3159. confirmText={t('settings.resetPreferences')}
  3160. variant="default"
  3161. onConfirm={() => {
  3162. setShowClearStorageConfirm(false);
  3163. localStorage.clear();
  3164. showToast(t('settings.toast.uiPreferencesReset'), 'success');
  3165. setTimeout(() => window.location.reload(), 1000);
  3166. }}
  3167. onCancel={() => setShowClearStorageConfirm(false)}
  3168. />
  3169. )}
  3170. {/* Confirm Modal: Bulk Plug Action */}
  3171. {showBulkPlugConfirm && (
  3172. <ConfirmModal
  3173. title={`Turn All Plugs ${showBulkPlugConfirm === 'on' ? 'On' : 'Off'}`}
  3174. message={`This will turn ${showBulkPlugConfirm === 'on' ? 'ON' : 'OFF'} all ${smartPlugs?.filter(p => p.enabled).length || 0} enabled smart plugs. ${showBulkPlugConfirm === 'off' ? 'Any running printers may be affected!' : ''}`}
  3175. confirmText={`Turn All ${showBulkPlugConfirm === 'on' ? 'On' : 'Off'}`}
  3176. variant={showBulkPlugConfirm === 'off' ? 'danger' : 'warning'}
  3177. onConfirm={() => {
  3178. const action = showBulkPlugConfirm;
  3179. setShowBulkPlugConfirm(null);
  3180. bulkPlugActionMutation.mutate(action);
  3181. }}
  3182. onCancel={() => setShowBulkPlugConfirm(null)}
  3183. />
  3184. )}
  3185. {/* Release Notes Modal */}
  3186. {showReleaseNotes && updateCheck?.release_notes && (
  3187. <div
  3188. className="fixed inset-0 bg-black/50 flex items-center justify-center z-50 p-4"
  3189. onClick={() => setShowReleaseNotes(false)}
  3190. >
  3191. <Card className="w-full max-w-2xl max-h-[80vh] flex flex-col" onClick={(e: React.MouseEvent) => e.stopPropagation()}>
  3192. <CardHeader className="flex flex-row items-center justify-between shrink-0">
  3193. <div>
  3194. <h2 className="text-lg font-semibold text-white">
  3195. Release Notes - v{updateCheck.latest_version}
  3196. </h2>
  3197. {updateCheck.release_name && updateCheck.release_name !== updateCheck.latest_version && (
  3198. <p className="text-sm text-bambu-gray">{updateCheck.release_name}</p>
  3199. )}
  3200. </div>
  3201. <button
  3202. onClick={() => setShowReleaseNotes(false)}
  3203. className="p-1 rounded hover:bg-bambu-dark-tertiary text-bambu-gray hover:text-white"
  3204. >
  3205. <X className="w-5 h-5" />
  3206. </button>
  3207. </CardHeader>
  3208. <CardContent className="overflow-y-auto flex-1">
  3209. <pre className="text-sm text-bambu-gray whitespace-pre-wrap font-sans">
  3210. {updateCheck.release_notes}
  3211. </pre>
  3212. </CardContent>
  3213. <div className="p-4 border-t border-bambu-dark-tertiary shrink-0 flex gap-2">
  3214. {updateCheck.release_url && (
  3215. <a
  3216. href={updateCheck.release_url}
  3217. target="_blank"
  3218. rel="noopener noreferrer"
  3219. className="flex-1"
  3220. >
  3221. <Button variant="secondary" className="w-full">
  3222. <ExternalLink className="w-4 h-4" />
  3223. View on GitHub
  3224. </Button>
  3225. </a>
  3226. )}
  3227. <Button
  3228. onClick={() => setShowReleaseNotes(false)}
  3229. className="flex-1"
  3230. >
  3231. Close
  3232. </Button>
  3233. </div>
  3234. </Card>
  3235. </div>
  3236. )}
  3237. {/* Users Tab */}
  3238. {activeTab === 'users' && (
  3239. <div className="space-y-6">
  3240. {/* Auth Toggle Header */}
  3241. <Card>
  3242. <CardContent className="py-4">
  3243. <div className="flex items-center justify-between">
  3244. <div className="flex items-center gap-3">
  3245. <div className={`w-10 h-10 rounded-full flex items-center justify-center ${authEnabled ? 'bg-green-500/20' : 'bg-gray-500/20'}`}>
  3246. {authEnabled ? (
  3247. <Lock className="w-5 h-5 text-green-400" />
  3248. ) : (
  3249. <Unlock className="w-5 h-5 text-gray-400" />
  3250. )}
  3251. </div>
  3252. <div>
  3253. <h3 className="text-white font-medium">{t('settings.authentication')}</h3>
  3254. <p className="text-sm text-bambu-gray">
  3255. {authEnabled
  3256. ? t('settings.authEnabledDescription')
  3257. : t('settings.authDisabledDescription')}
  3258. </p>
  3259. </div>
  3260. </div>
  3261. {!authEnabled ? (
  3262. <Button onClick={() => navigate('/setup')}>
  3263. <Lock className="w-4 h-4" />
  3264. {t('common.enable')}
  3265. </Button>
  3266. ) : user?.is_admin && (
  3267. <Button variant="secondary" onClick={() => setShowDisableAuthConfirm(true)}>
  3268. <Unlock className="w-4 h-4" />
  3269. {t('common.disable')}
  3270. </Button>
  3271. )}
  3272. </div>
  3273. </CardContent>
  3274. </Card>
  3275. {authEnabled && (
  3276. <div className="grid grid-cols-1 xl:grid-cols-2 gap-6">
  3277. {/* Left Column: Current User + User List */}
  3278. <div className="space-y-6">
  3279. {/* Current User Card */}
  3280. {user && (
  3281. <Card>
  3282. <CardHeader>
  3283. <div className="flex items-center justify-between">
  3284. <h3 className="text-lg font-semibold text-white flex items-center gap-2">
  3285. <Users className="w-5 h-5 text-bambu-green" />
  3286. {t('settings.currentUser')}
  3287. </h3>
  3288. <Button size="sm" variant="ghost" onClick={() => setShowChangePasswordModal(true)}>
  3289. <Key className="w-4 h-4" />
  3290. {t('settings.changePassword')}
  3291. </Button>
  3292. </div>
  3293. </CardHeader>
  3294. <CardContent>
  3295. <div className="flex items-center justify-between">
  3296. <div>
  3297. <p className="text-white font-medium text-lg">{user.username}</p>
  3298. <div className="flex flex-wrap gap-1 mt-2">
  3299. {user.is_admin && (
  3300. <span className="px-2 py-0.5 rounded-full text-xs font-medium bg-purple-500/20 text-purple-300">
  3301. {t('settings.admin')}
  3302. </span>
  3303. )}
  3304. {user.groups?.map(group => (
  3305. <span
  3306. key={group.id}
  3307. className={`px-2 py-0.5 rounded-full text-xs font-medium ${
  3308. group.name === 'Administrators'
  3309. ? 'bg-purple-500/20 text-purple-300'
  3310. : group.name === 'Operators'
  3311. ? 'bg-blue-500/20 text-blue-300'
  3312. : group.name === 'Viewers'
  3313. ? 'bg-green-500/20 text-green-300'
  3314. : 'bg-gray-500/20 text-gray-300'
  3315. }`}
  3316. >
  3317. {group.name}
  3318. </span>
  3319. ))}
  3320. </div>
  3321. </div>
  3322. </div>
  3323. </CardContent>
  3324. </Card>
  3325. )}
  3326. {/* User List */}
  3327. <Card>
  3328. <CardHeader>
  3329. <div className="flex items-center justify-between">
  3330. <h3 className="text-lg font-semibold text-white flex items-center gap-2">
  3331. <Users className="w-5 h-5 text-bambu-green" />
  3332. {t('settings.users')}
  3333. </h3>
  3334. {hasPermission('users:create') && (
  3335. <Button
  3336. size="sm"
  3337. onClick={() => {
  3338. setShowCreateUserModal(true);
  3339. setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
  3340. }}
  3341. >
  3342. <Plus className="w-4 h-4" />
  3343. {t('settings.addUser')}
  3344. </Button>
  3345. )}
  3346. </div>
  3347. </CardHeader>
  3348. <CardContent>
  3349. {usersLoading ? (
  3350. <div className="flex items-center justify-center py-8">
  3351. <Loader2 className="w-6 h-6 text-bambu-green animate-spin" />
  3352. </div>
  3353. ) : usersData.length === 0 ? (
  3354. <p className="text-center text-bambu-gray py-8">{t('settings.noUsersFound')}</p>
  3355. ) : (
  3356. <div className="divide-y divide-bambu-dark-tertiary">
  3357. {usersData.map((userItem) => (
  3358. <div key={userItem.id} className="py-3 flex items-center justify-between">
  3359. <div className="flex-1 min-w-0">
  3360. <p className="text-white font-medium truncate">{userItem.username}</p>
  3361. <div className="flex flex-wrap gap-1 mt-1">
  3362. {userItem.is_admin && (
  3363. <span className="px-2 py-0.5 rounded-full text-xs font-medium bg-purple-500/20 text-purple-300">
  3364. {t('settings.admin')}
  3365. </span>
  3366. )}
  3367. {userItem.groups?.map(group => (
  3368. <span
  3369. key={group.id}
  3370. className={`px-2 py-0.5 rounded-full text-xs font-medium ${
  3371. group.name === 'Administrators'
  3372. ? 'bg-purple-500/20 text-purple-300'
  3373. : group.name === 'Operators'
  3374. ? 'bg-blue-500/20 text-blue-300'
  3375. : group.name === 'Viewers'
  3376. ? 'bg-green-500/20 text-green-300'
  3377. : 'bg-gray-500/20 text-gray-300'
  3378. }`}
  3379. >
  3380. {group.name}
  3381. </span>
  3382. ))}
  3383. </div>
  3384. </div>
  3385. <div className="flex items-center gap-1 ml-4">
  3386. {hasPermission('users:update') && (
  3387. <Button size="sm" variant="ghost" onClick={() => startEditUser(userItem)}>
  3388. <Edit2 className="w-4 h-4" />
  3389. </Button>
  3390. )}
  3391. {hasPermission('users:delete') && userItem.id !== user?.id && (
  3392. <Button size="sm" variant="ghost" onClick={() => handleDeleteUserClick(userItem.id)}>
  3393. <Trash2 className="w-4 h-4" />
  3394. </Button>
  3395. )}
  3396. </div>
  3397. </div>
  3398. ))}
  3399. </div>
  3400. )}
  3401. </CardContent>
  3402. </Card>
  3403. </div>
  3404. {/* Right Column: Groups */}
  3405. <div>
  3406. <Card>
  3407. <CardHeader>
  3408. <div className="flex items-center justify-between">
  3409. <h3 className="text-lg font-semibold text-white flex items-center gap-2">
  3410. <Shield className="w-5 h-5 text-bambu-green" />
  3411. {t('settings.groups')}
  3412. </h3>
  3413. {hasPermission('groups:create') && (
  3414. <Button
  3415. size="sm"
  3416. onClick={() => {
  3417. setShowCreateGroupModal(true);
  3418. resetGroupForm();
  3419. }}
  3420. >
  3421. <Plus className="w-4 h-4" />
  3422. {t('settings.addGroup')}
  3423. </Button>
  3424. )}
  3425. </div>
  3426. </CardHeader>
  3427. <CardContent>
  3428. {groupsLoading ? (
  3429. <div className="flex items-center justify-center py-8">
  3430. <Loader2 className="w-6 h-6 text-bambu-green animate-spin" />
  3431. </div>
  3432. ) : groupsData.length === 0 ? (
  3433. <p className="text-center text-bambu-gray py-8">{t('settings.noGroupsFound')}</p>
  3434. ) : (
  3435. <div className="divide-y divide-bambu-dark-tertiary">
  3436. {groupsData.map((group) => (
  3437. <div key={group.id} className="py-3">
  3438. <div className="flex items-center justify-between">
  3439. <div className="flex items-center gap-2">
  3440. <Shield
  3441. className={`w-4 h-4 ${
  3442. group.name === 'Administrators'
  3443. ? 'text-purple-400'
  3444. : group.name === 'Operators'
  3445. ? 'text-blue-400'
  3446. : group.name === 'Viewers'
  3447. ? 'text-green-400'
  3448. : 'text-bambu-gray'
  3449. }`}
  3450. />
  3451. <span className="text-white font-medium">{group.name}</span>
  3452. {group.is_system && (
  3453. <span className="px-2 py-0.5 rounded text-xs bg-yellow-500/20 text-yellow-400">
  3454. {t('settings.system')}
  3455. </span>
  3456. )}
  3457. </div>
  3458. <div className="flex items-center gap-1">
  3459. {hasPermission('groups:update') && (
  3460. <Button size="sm" variant="ghost" onClick={() => startEditGroup(group)}>
  3461. <Edit2 className="w-4 h-4" />
  3462. </Button>
  3463. )}
  3464. {hasPermission('groups:delete') && !group.is_system && (
  3465. <Button size="sm" variant="ghost" onClick={() => setDeleteGroupId(group.id)}>
  3466. <Trash2 className="w-4 h-4" />
  3467. </Button>
  3468. )}
  3469. </div>
  3470. </div>
  3471. <p className="text-sm text-bambu-gray mt-1 ml-6">
  3472. {group.description || t('settings.noDescription')}
  3473. </p>
  3474. <div className="flex items-center gap-4 mt-2 ml-6 text-xs text-bambu-gray">
  3475. <span>{t('settings.userCount', { count: group.user_count })}</span>
  3476. <span>{t('settings.permissionCount', { count: group.permissions.length })}</span>
  3477. </div>
  3478. </div>
  3479. ))}
  3480. </div>
  3481. )}
  3482. </CardContent>
  3483. </Card>
  3484. </div>
  3485. </div>
  3486. )}
  3487. {/* Auth Disabled Info */}
  3488. {!authEnabled && (
  3489. <Card>
  3490. <CardContent className="py-6">
  3491. <div className="text-center">
  3492. <Unlock className="w-12 h-12 text-bambu-gray mx-auto mb-4" />
  3493. <h3 className="text-lg font-medium text-white mb-2">{t('settings.authDisabledTitle')}</h3>
  3494. <p className="text-sm text-bambu-gray mb-4 max-w-md mx-auto">
  3495. {t('settings.authDisabledMessage')}
  3496. </p>
  3497. <ul className="space-y-2 text-sm text-bambu-gray mb-6 text-left max-w-xs mx-auto">
  3498. <li className="flex items-start gap-2">
  3499. <CheckCircle className="w-4 h-4 text-bambu-green mt-0.5 flex-shrink-0" />
  3500. <span>{t('settings.authDisabledFeature1')}</span>
  3501. </li>
  3502. <li className="flex items-start gap-2">
  3503. <CheckCircle className="w-4 h-4 text-bambu-green mt-0.5 flex-shrink-0" />
  3504. <span>{t('settings.authDisabledFeature2')}</span>
  3505. </li>
  3506. <li className="flex items-start gap-2">
  3507. <CheckCircle className="w-4 h-4 text-bambu-green mt-0.5 flex-shrink-0" />
  3508. <span>{t('settings.authDisabledFeature3')}</span>
  3509. </li>
  3510. </ul>
  3511. <Button onClick={() => navigate('/setup')}>
  3512. <Lock className="w-4 h-4" />
  3513. {t('settings.enableAuthentication')}
  3514. </Button>
  3515. </div>
  3516. </CardContent>
  3517. </Card>
  3518. )}
  3519. </div>
  3520. )}
  3521. {/* Create User Modal */}
  3522. {showCreateUserModal && (
  3523. <div
  3524. className="fixed inset-0 bg-black flex items-center justify-center z-50 p-4"
  3525. onClick={() => {
  3526. setShowCreateUserModal(false);
  3527. setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
  3528. }}
  3529. >
  3530. <Card
  3531. className="w-full max-w-md"
  3532. onClick={(e: React.MouseEvent) => e.stopPropagation()}
  3533. >
  3534. <CardHeader>
  3535. <div className="flex items-center justify-between">
  3536. <div className="flex items-center gap-2">
  3537. <Users className="w-5 h-5 text-bambu-green" />
  3538. <h2 className="text-lg font-semibold text-white">{t('settings.createUser')}</h2>
  3539. </div>
  3540. <Button
  3541. variant="ghost"
  3542. size="sm"
  3543. onClick={() => {
  3544. setShowCreateUserModal(false);
  3545. setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
  3546. }}
  3547. >
  3548. <X className="w-5 h-5" />
  3549. </Button>
  3550. </div>
  3551. </CardHeader>
  3552. <CardContent>
  3553. <div className="space-y-4">
  3554. <div>
  3555. <label className="block text-sm font-medium text-white mb-2">{t('settings.username')}</label>
  3556. <input
  3557. type="text"
  3558. value={userFormData.username}
  3559. onChange={(e) => setUserFormData({ ...userFormData, username: e.target.value })}
  3560. className="w-full px-4 py-3 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg text-white placeholder-bambu-gray focus:outline-none focus:ring-2 focus:ring-bambu-green/50 focus:border-bambu-green transition-colors"
  3561. placeholder={t('settings.enterUsername')}
  3562. autoComplete="username"
  3563. />
  3564. </div>
  3565. <div>
  3566. <label className="block text-sm font-medium text-white mb-2">{t('settings.password')}</label>
  3567. <input
  3568. type="password"
  3569. value={userFormData.password}
  3570. onChange={(e) => setUserFormData({ ...userFormData, password: e.target.value })}
  3571. className="w-full px-4 py-3 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg text-white placeholder-bambu-gray focus:outline-none focus:ring-2 focus:ring-bambu-green/50 focus:border-bambu-green transition-colors"
  3572. placeholder={t('settings.enterPassword')}
  3573. autoComplete="new-password"
  3574. minLength={6}
  3575. />
  3576. </div>
  3577. <div>
  3578. <label className="block text-sm font-medium text-white mb-2">{t('settings.confirmPassword')}</label>
  3579. <input
  3580. type="password"
  3581. value={userFormData.confirmPassword}
  3582. onChange={(e) => setUserFormData({ ...userFormData, confirmPassword: e.target.value })}
  3583. className={`w-full px-4 py-3 bg-bambu-dark-secondary border rounded-lg text-white placeholder-bambu-gray focus:outline-none focus:ring-2 focus:ring-bambu-green/50 focus:border-bambu-green transition-colors ${
  3584. userFormData.confirmPassword && userFormData.password !== userFormData.confirmPassword
  3585. ? 'border-red-500'
  3586. : 'border-bambu-dark-tertiary'
  3587. }`}
  3588. placeholder={t('settings.confirmPasswordPlaceholder')}
  3589. autoComplete="new-password"
  3590. minLength={6}
  3591. />
  3592. {userFormData.confirmPassword && userFormData.password !== userFormData.confirmPassword && (
  3593. <p className="text-red-400 text-xs mt-1">{t('settings.passwordsDoNotMatch')}</p>
  3594. )}
  3595. </div>
  3596. <div>
  3597. <label className="block text-sm font-medium text-white mb-2">Groups</label>
  3598. <div className="space-y-2 max-h-40 overflow-y-auto p-2 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg">
  3599. {groupsData.map(group => (
  3600. <label
  3601. key={group.id}
  3602. className="flex items-center gap-3 px-2 py-1.5 rounded hover:bg-bambu-dark-tertiary cursor-pointer"
  3603. >
  3604. <input
  3605. type="checkbox"
  3606. checked={userFormData.group_ids.includes(group.id)}
  3607. onChange={() => toggleUserGroup(group.id)}
  3608. className="w-4 h-4 rounded border-bambu-gray text-bambu-green focus:ring-bambu-green focus:ring-offset-0 bg-bambu-dark"
  3609. />
  3610. <span className="text-sm text-white">{group.name}</span>
  3611. {group.is_system && (
  3612. <span className="text-xs text-yellow-400">(System)</span>
  3613. )}
  3614. </label>
  3615. ))}
  3616. {groupsData.length === 0 && (
  3617. <p className="text-sm text-bambu-gray">{t('settings.noGroupsAvailable')}</p>
  3618. )}
  3619. </div>
  3620. </div>
  3621. </div>
  3622. <div className="mt-6 flex justify-end gap-3">
  3623. <Button
  3624. variant="secondary"
  3625. onClick={() => {
  3626. setShowCreateUserModal(false);
  3627. setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
  3628. }}
  3629. >
  3630. Cancel
  3631. </Button>
  3632. <Button
  3633. onClick={handleCreateUser}
  3634. disabled={createUserMutation.isPending || !userFormData.username || !userFormData.password || userFormData.password !== userFormData.confirmPassword || userFormData.password.length < 6}
  3635. >
  3636. {createUserMutation.isPending ? (
  3637. <>
  3638. <Loader2 className="w-4 h-4 animate-spin" />
  3639. Creating...
  3640. </>
  3641. ) : (
  3642. <>
  3643. <Plus className="w-4 h-4" />
  3644. Create User
  3645. </>
  3646. )}
  3647. </Button>
  3648. </div>
  3649. </CardContent>
  3650. </Card>
  3651. </div>
  3652. )}
  3653. {/* Edit User Modal */}
  3654. {showEditUserModal && editingUserId !== null && (
  3655. <div
  3656. className="fixed inset-0 bg-black flex items-center justify-center z-50 p-4"
  3657. onClick={() => {
  3658. setShowEditUserModal(false);
  3659. setEditingUserId(null);
  3660. setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
  3661. }}
  3662. >
  3663. <Card
  3664. className="w-full max-w-md"
  3665. onClick={(e: React.MouseEvent) => e.stopPropagation()}
  3666. >
  3667. <CardHeader>
  3668. <div className="flex items-center justify-between">
  3669. <div className="flex items-center gap-2">
  3670. <Edit2 className="w-5 h-5 text-bambu-green" />
  3671. <h2 className="text-lg font-semibold text-white">{t('settings.editUser')}</h2>
  3672. </div>
  3673. <Button
  3674. variant="ghost"
  3675. size="sm"
  3676. onClick={() => {
  3677. setShowEditUserModal(false);
  3678. setEditingUserId(null);
  3679. setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
  3680. }}
  3681. >
  3682. <X className="w-5 h-5" />
  3683. </Button>
  3684. </div>
  3685. </CardHeader>
  3686. <CardContent>
  3687. <div className="space-y-4">
  3688. <div>
  3689. <label className="block text-sm font-medium text-white mb-2">{t('settings.username')}</label>
  3690. <input
  3691. type="text"
  3692. value={userFormData.username}
  3693. onChange={(e) => setUserFormData({ ...userFormData, username: e.target.value })}
  3694. className="w-full px-4 py-3 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg text-white placeholder-bambu-gray focus:outline-none focus:ring-2 focus:ring-bambu-green/50 focus:border-bambu-green transition-colors"
  3695. placeholder={t('settings.enterUsername')}
  3696. autoComplete="username"
  3697. />
  3698. </div>
  3699. <div>
  3700. <label className="block text-sm font-medium text-white mb-2">
  3701. Password <span className="text-bambu-gray font-normal">(leave blank to keep current)</span>
  3702. </label>
  3703. <input
  3704. type="password"
  3705. value={userFormData.password}
  3706. onChange={(e) => setUserFormData({ ...userFormData, password: e.target.value, confirmPassword: '' })}
  3707. className="w-full px-4 py-3 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg text-white placeholder-bambu-gray focus:outline-none focus:ring-2 focus:ring-bambu-green/50 focus:border-bambu-green transition-colors"
  3708. placeholder={t('settings.enterNewPassword')}
  3709. autoComplete="new-password"
  3710. minLength={6}
  3711. />
  3712. </div>
  3713. {userFormData.password && (
  3714. <div>
  3715. <label className="block text-sm font-medium text-white mb-2">{t('settings.confirmPassword')}</label>
  3716. <input
  3717. type="password"
  3718. value={userFormData.confirmPassword}
  3719. onChange={(e) => setUserFormData({ ...userFormData, confirmPassword: e.target.value })}
  3720. className={`w-full px-4 py-3 bg-bambu-dark-secondary border rounded-lg text-white placeholder-bambu-gray focus:outline-none focus:ring-2 focus:ring-bambu-green/50 focus:border-bambu-green transition-colors ${
  3721. userFormData.confirmPassword && userFormData.password !== userFormData.confirmPassword
  3722. ? 'border-red-500'
  3723. : 'border-bambu-dark-tertiary'
  3724. }`}
  3725. placeholder={t('settings.confirmNewPassword')}
  3726. autoComplete="new-password"
  3727. minLength={6}
  3728. />
  3729. {userFormData.confirmPassword && userFormData.password !== userFormData.confirmPassword && (
  3730. <p className="text-red-400 text-xs mt-1">{t('settings.passwordsDoNotMatch')}</p>
  3731. )}
  3732. </div>
  3733. )}
  3734. <div>
  3735. <label className="block text-sm font-medium text-white mb-2">Groups</label>
  3736. <div className="space-y-2 max-h-40 overflow-y-auto p-2 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg">
  3737. {groupsData.map(group => (
  3738. <label
  3739. key={group.id}
  3740. className="flex items-center gap-3 px-2 py-1.5 rounded hover:bg-bambu-dark-tertiary cursor-pointer"
  3741. >
  3742. <input
  3743. type="checkbox"
  3744. checked={userFormData.group_ids.includes(group.id)}
  3745. onChange={() => toggleUserGroup(group.id)}
  3746. className="w-4 h-4 rounded border-bambu-gray text-bambu-green focus:ring-bambu-green focus:ring-offset-0 bg-bambu-dark"
  3747. />
  3748. <span className="text-sm text-white">{group.name}</span>
  3749. {group.is_system && (
  3750. <span className="text-xs text-yellow-400">(System)</span>
  3751. )}
  3752. </label>
  3753. ))}
  3754. </div>
  3755. </div>
  3756. </div>
  3757. <div className="mt-6 flex justify-end gap-3">
  3758. <Button
  3759. variant="secondary"
  3760. onClick={() => {
  3761. setShowEditUserModal(false);
  3762. setEditingUserId(null);
  3763. setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
  3764. }}
  3765. >
  3766. Cancel
  3767. </Button>
  3768. <Button
  3769. onClick={() => handleUpdateUser(editingUserId)}
  3770. disabled={updateUserMutation.isPending || !userFormData.username || !!(userFormData.password && (userFormData.password !== userFormData.confirmPassword || userFormData.password.length < 6))}
  3771. >
  3772. {updateUserMutation.isPending ? (
  3773. <>
  3774. <Loader2 className="w-4 h-4 animate-spin" />
  3775. Saving...
  3776. </>
  3777. ) : (
  3778. <>
  3779. <Save className="w-4 h-4" />
  3780. Save Changes
  3781. </>
  3782. )}
  3783. </Button>
  3784. </div>
  3785. </CardContent>
  3786. </Card>
  3787. </div>
  3788. )}
  3789. {/* Delete User Confirmation Modal */}
  3790. {deleteUserId !== null && (
  3791. <div
  3792. className="fixed inset-0 bg-black/80 flex items-center justify-center z-50 p-4"
  3793. onClick={() => {
  3794. setDeleteUserId(null);
  3795. setDeleteUserItemCounts(null);
  3796. }}
  3797. >
  3798. <Card
  3799. className="w-full max-w-md"
  3800. onClick={(e: React.MouseEvent) => e.stopPropagation()}
  3801. >
  3802. <CardHeader>
  3803. <div className="flex items-center gap-2 text-red-400">
  3804. <Trash2 className="w-5 h-5" />
  3805. <h3 className="text-lg font-semibold">{t('settings.deleteUserTitle')}</h3>
  3806. </div>
  3807. </CardHeader>
  3808. <CardContent className="space-y-4">
  3809. {deleteUserLoading ? (
  3810. <div className="flex items-center justify-center py-4">
  3811. <div className="animate-spin rounded-full h-6 w-6 border-2 border-bambu-green border-t-transparent" />
  3812. </div>
  3813. ) : deleteUserItemCounts && (deleteUserItemCounts.archives + deleteUserItemCounts.queue_items + deleteUserItemCounts.library_files > 0) ? (
  3814. <>
  3815. <p className="text-white">{t('settings.userHasCreated')}</p>
  3816. <ul className="list-disc list-inside text-bambu-gray space-y-1">
  3817. {deleteUserItemCounts.archives > 0 && (
  3818. <li>{deleteUserItemCounts.archives} archive{deleteUserItemCounts.archives !== 1 ? 's' : ''}</li>
  3819. )}
  3820. {deleteUserItemCounts.queue_items > 0 && (
  3821. <li>{deleteUserItemCounts.queue_items} queue item{deleteUserItemCounts.queue_items !== 1 ? 's' : ''}</li>
  3822. )}
  3823. {deleteUserItemCounts.library_files > 0 && (
  3824. <li>{deleteUserItemCounts.library_files} library file{deleteUserItemCounts.library_files !== 1 ? 's' : ''}</li>
  3825. )}
  3826. </ul>
  3827. <p className="text-bambu-gray text-sm">{t('settings.userItemsQuestion')}</p>
  3828. <div className="flex flex-col gap-2">
  3829. <Button
  3830. variant="danger"
  3831. onClick={() => deleteUserMutation.mutate({ id: deleteUserId, deleteItems: true })}
  3832. disabled={deleteUserMutation.isPending}
  3833. className="justify-center"
  3834. >
  3835. Delete user AND their items
  3836. </Button>
  3837. <Button
  3838. variant="secondary"
  3839. onClick={() => deleteUserMutation.mutate({ id: deleteUserId, deleteItems: false })}
  3840. disabled={deleteUserMutation.isPending}
  3841. className="justify-center"
  3842. >
  3843. Delete user, keep items (become ownerless)
  3844. </Button>
  3845. <Button
  3846. variant="ghost"
  3847. onClick={() => {
  3848. setDeleteUserId(null);
  3849. setDeleteUserItemCounts(null);
  3850. }}
  3851. disabled={deleteUserMutation.isPending}
  3852. className="justify-center"
  3853. >
  3854. Cancel
  3855. </Button>
  3856. </div>
  3857. </>
  3858. ) : (
  3859. <>
  3860. <p className="text-white">{t('settings.deleteUserConfirm')}</p>
  3861. <p className="text-bambu-gray text-sm">{t('settings.actionCannotBeUndone')}</p>
  3862. <div className="flex gap-2 justify-end">
  3863. <Button
  3864. variant="ghost"
  3865. onClick={() => {
  3866. setDeleteUserId(null);
  3867. setDeleteUserItemCounts(null);
  3868. }}
  3869. disabled={deleteUserMutation.isPending}
  3870. >
  3871. Cancel
  3872. </Button>
  3873. <Button
  3874. variant="danger"
  3875. onClick={() => deleteUserMutation.mutate({ id: deleteUserId, deleteItems: false })}
  3876. disabled={deleteUserMutation.isPending}
  3877. >
  3878. Delete User
  3879. </Button>
  3880. </div>
  3881. </>
  3882. )}
  3883. </CardContent>
  3884. </Card>
  3885. </div>
  3886. )}
  3887. {/* Create/Edit Group Modal */}
  3888. {(showCreateGroupModal || editingGroup) && (
  3889. <div
  3890. className="fixed inset-0 bg-black flex items-center justify-center z-50 p-4"
  3891. onClick={() => {
  3892. setShowCreateGroupModal(false);
  3893. setEditingGroup(null);
  3894. resetGroupForm();
  3895. }}
  3896. >
  3897. <Card
  3898. className="w-full max-w-2xl max-h-[90vh] overflow-y-auto"
  3899. onClick={(e: React.MouseEvent) => e.stopPropagation()}
  3900. >
  3901. <CardHeader>
  3902. <div className="flex items-center justify-between">
  3903. <div className="flex items-center gap-2">
  3904. <Shield className="w-5 h-5 text-bambu-green" />
  3905. <h2 className="text-lg font-semibold text-white">
  3906. {editingGroup ? 'Edit Group' : 'Create Group'}
  3907. </h2>
  3908. </div>
  3909. <Button
  3910. variant="ghost"
  3911. size="sm"
  3912. onClick={() => {
  3913. setShowCreateGroupModal(false);
  3914. setEditingGroup(null);
  3915. resetGroupForm();
  3916. }}
  3917. >
  3918. <X className="w-5 h-5" />
  3919. </Button>
  3920. </div>
  3921. </CardHeader>
  3922. <CardContent>
  3923. <div className="space-y-4">
  3924. <div>
  3925. <label className="block text-sm font-medium text-white mb-2">{t('settings.groupName')}</label>
  3926. <input
  3927. type="text"
  3928. value={groupFormData.name}
  3929. onChange={(e) => setGroupFormData({ ...groupFormData, name: e.target.value })}
  3930. disabled={editingGroup?.is_system}
  3931. className="w-full px-4 py-3 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg text-white placeholder-bambu-gray focus:outline-none focus:ring-2 focus:ring-bambu-green/50 focus:border-bambu-green transition-colors disabled:opacity-50"
  3932. placeholder={t('settings.enterGroupName')}
  3933. />
  3934. {editingGroup?.is_system && (
  3935. <p className="text-xs text-yellow-400 mt-1">{t('settings.systemGroupWarning')}</p>
  3936. )}
  3937. </div>
  3938. <div>
  3939. <label className="block text-sm font-medium text-white mb-2">Description</label>
  3940. <textarea
  3941. value={groupFormData.description}
  3942. onChange={(e) => setGroupFormData({ ...groupFormData, description: e.target.value })}
  3943. rows={2}
  3944. className="w-full px-4 py-3 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg text-white placeholder-bambu-gray focus:outline-none focus:ring-2 focus:ring-bambu-green/50 focus:border-bambu-green transition-colors resize-none"
  3945. placeholder={t('settings.enterDescriptionOptional')}
  3946. />
  3947. </div>
  3948. <div>
  3949. <label className="block text-sm font-medium text-white mb-2">
  3950. Permissions ({groupFormData.permissions.length} selected)
  3951. </label>
  3952. <div className="space-y-2 max-h-96 overflow-y-auto">
  3953. {permissionsData?.categories.map((category) => (
  3954. <div key={category.name} className="border border-bambu-dark-tertiary rounded-lg overflow-hidden">
  3955. <div
  3956. className="flex items-center justify-between px-4 py-2 bg-bambu-dark-secondary cursor-pointer hover:bg-bambu-dark-tertiary transition-colors"
  3957. onClick={() => toggleCategory(category.name)}
  3958. >
  3959. <div className="flex items-center gap-3">
  3960. <button
  3961. type="button"
  3962. onClick={(e) => {
  3963. e.stopPropagation();
  3964. toggleCategoryPermissions(category, !isCategoryFullySelected(category));
  3965. }}
  3966. className={`w-5 h-5 rounded border flex items-center justify-center transition-colors ${
  3967. isCategoryFullySelected(category)
  3968. ? 'bg-bambu-green border-bambu-green'
  3969. : isCategoryPartiallySelected(category)
  3970. ? 'bg-bambu-green/50 border-bambu-green'
  3971. : 'border-bambu-gray hover:border-white'
  3972. }`}
  3973. >
  3974. {(isCategoryFullySelected(category) || isCategoryPartiallySelected(category)) && (
  3975. <Check className="w-3 h-3 text-white" />
  3976. )}
  3977. </button>
  3978. <span className="text-white font-medium">{category.name}</span>
  3979. <span className="text-xs text-bambu-gray">
  3980. ({category.permissions.filter((p) => groupFormData.permissions.includes(p.value)).length}/
  3981. {category.permissions.length})
  3982. </span>
  3983. </div>
  3984. {expandedCategories.has(category.name) ? (
  3985. <ChevronDown className="w-4 h-4 text-bambu-gray" />
  3986. ) : (
  3987. <ChevronRight className="w-4 h-4 text-bambu-gray" />
  3988. )}
  3989. </div>
  3990. {expandedCategories.has(category.name) && (
  3991. <div className="p-3 bg-bambu-dark space-y-2">
  3992. {category.permissions.map((perm) => (
  3993. <label
  3994. key={perm.value}
  3995. className="flex items-center gap-3 px-2 py-1.5 rounded hover:bg-bambu-dark-secondary cursor-pointer"
  3996. >
  3997. <input
  3998. type="checkbox"
  3999. checked={groupFormData.permissions.includes(perm.value)}
  4000. onChange={() => togglePermission(perm.value)}
  4001. className="w-4 h-4 rounded border-bambu-gray text-bambu-green focus:ring-bambu-green focus:ring-offset-0 bg-bambu-dark-secondary"
  4002. />
  4003. <span className="text-sm text-bambu-gray">{perm.label}</span>
  4004. </label>
  4005. ))}
  4006. </div>
  4007. )}
  4008. </div>
  4009. ))}
  4010. </div>
  4011. </div>
  4012. </div>
  4013. <div className="mt-6 flex justify-end gap-3">
  4014. <Button
  4015. variant="secondary"
  4016. onClick={() => {
  4017. setShowCreateGroupModal(false);
  4018. setEditingGroup(null);
  4019. resetGroupForm();
  4020. }}
  4021. >
  4022. Cancel
  4023. </Button>
  4024. <Button
  4025. onClick={editingGroup ? handleUpdateGroup : handleCreateGroup}
  4026. disabled={createGroupMutation.isPending || updateGroupMutation.isPending || !groupFormData.name.trim()}
  4027. >
  4028. {(createGroupMutation.isPending || updateGroupMutation.isPending) ? (
  4029. <>
  4030. <Loader2 className="w-4 h-4 animate-spin" />
  4031. {editingGroup ? 'Saving...' : 'Creating...'}
  4032. </>
  4033. ) : (
  4034. <>
  4035. <Save className="w-4 h-4" />
  4036. {editingGroup ? 'Save Changes' : 'Create Group'}
  4037. </>
  4038. )}
  4039. </Button>
  4040. </div>
  4041. </CardContent>
  4042. </Card>
  4043. </div>
  4044. )}
  4045. {/* Delete Group Confirmation Modal */}
  4046. {deleteGroupId !== null && (
  4047. <ConfirmModal
  4048. title={t('settings.deleteGroupTitle')}
  4049. message={t('settings.deleteGroupMessage')}
  4050. confirmText={t('settings.deleteGroup')}
  4051. variant="danger"
  4052. onConfirm={() => {
  4053. deleteGroupMutation.mutate(deleteGroupId);
  4054. setDeleteGroupId(null);
  4055. }}
  4056. onCancel={() => setDeleteGroupId(null)}
  4057. />
  4058. )}
  4059. {/* Backup Tab */}
  4060. {activeTab === 'backup' && (
  4061. <GitHubBackupSettings />
  4062. )}
  4063. {/* Disable Authentication Confirmation Modal */}
  4064. {showDisableAuthConfirm && (
  4065. <ConfirmModal
  4066. title={t('settings.disableAuthenticationTitle')}
  4067. message={t('settings.disableAuthenticationMessage')}
  4068. confirmText={t('settings.disableAuthentication')}
  4069. variant="danger"
  4070. onConfirm={async () => {
  4071. try {
  4072. await api.disableAuth();
  4073. showToast(t('settings.toast.authDisabled'), 'success');
  4074. await refreshAuth();
  4075. setShowDisableAuthConfirm(false);
  4076. // Refresh the page to ensure all protected routes are accessible
  4077. window.location.href = '/';
  4078. } catch (error: unknown) {
  4079. const message = error instanceof Error ? error.message : t('settings.toast.authDisableFailed');
  4080. showToast(message, 'error');
  4081. }
  4082. }}
  4083. onCancel={() => setShowDisableAuthConfirm(false)}
  4084. />
  4085. )}
  4086. {/* Change Password Modal */}
  4087. {showChangePasswordModal && (
  4088. <div
  4089. className="fixed inset-0 bg-black/70 flex items-center justify-center z-50 p-4"
  4090. onClick={() => {
  4091. setShowChangePasswordModal(false);
  4092. setChangePasswordData({ currentPassword: '', newPassword: '', confirmPassword: '' });
  4093. }}
  4094. >
  4095. <Card
  4096. className="w-full max-w-md"
  4097. onClick={(e: React.MouseEvent) => e.stopPropagation()}
  4098. >
  4099. <CardHeader>
  4100. <div className="flex items-center justify-between">
  4101. <div className="flex items-center gap-2">
  4102. <Key className="w-5 h-5 text-bambu-green" />
  4103. <h2 className="text-lg font-semibold text-white">{t('settings.changePassword')}</h2>
  4104. </div>
  4105. <Button
  4106. variant="ghost"
  4107. size="sm"
  4108. onClick={() => {
  4109. setShowChangePasswordModal(false);
  4110. setChangePasswordData({ currentPassword: '', newPassword: '', confirmPassword: '' });
  4111. }}
  4112. >
  4113. <X className="w-5 h-5" />
  4114. </Button>
  4115. </div>
  4116. </CardHeader>
  4117. <CardContent>
  4118. <div className="space-y-4">
  4119. <div>
  4120. <label className="block text-sm font-medium text-white mb-2">
  4121. Current Password
  4122. </label>
  4123. <input
  4124. type="password"
  4125. value={changePasswordData.currentPassword}
  4126. onChange={(e) => setChangePasswordData({ ...changePasswordData, currentPassword: e.target.value })}
  4127. className="w-full px-4 py-3 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg text-white placeholder-bambu-gray focus:outline-none focus:ring-2 focus:ring-bambu-green/50 focus:border-bambu-green transition-colors"
  4128. placeholder={t('settings.enterCurrentPassword')}
  4129. autoComplete="current-password"
  4130. />
  4131. </div>
  4132. <div>
  4133. <label className="block text-sm font-medium text-white mb-2">
  4134. New Password
  4135. </label>
  4136. <input
  4137. type="password"
  4138. value={changePasswordData.newPassword}
  4139. onChange={(e) => setChangePasswordData({ ...changePasswordData, newPassword: e.target.value })}
  4140. className="w-full px-4 py-3 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg text-white placeholder-bambu-gray focus:outline-none focus:ring-2 focus:ring-bambu-green/50 focus:border-bambu-green transition-colors"
  4141. placeholder={t('settings.enterNewPasswordMin6')}
  4142. autoComplete="new-password"
  4143. minLength={6}
  4144. />
  4145. </div>
  4146. <div>
  4147. <label className="block text-sm font-medium text-white mb-2">
  4148. Confirm New Password
  4149. </label>
  4150. <input
  4151. type="password"
  4152. value={changePasswordData.confirmPassword}
  4153. onChange={(e) => setChangePasswordData({ ...changePasswordData, confirmPassword: e.target.value })}
  4154. className={`w-full px-4 py-3 bg-bambu-dark-secondary border rounded-lg text-white placeholder-bambu-gray focus:outline-none focus:ring-2 focus:ring-bambu-green/50 focus:border-bambu-green transition-colors ${
  4155. changePasswordData.confirmPassword && changePasswordData.newPassword !== changePasswordData.confirmPassword
  4156. ? 'border-red-500'
  4157. : 'border-bambu-dark-tertiary'
  4158. }`}
  4159. placeholder={t('settings.confirmNewPassword')}
  4160. autoComplete="new-password"
  4161. minLength={6}
  4162. />
  4163. {changePasswordData.confirmPassword && changePasswordData.newPassword !== changePasswordData.confirmPassword && (
  4164. <p className="text-red-400 text-xs mt-1">{t('settings.passwordsDoNotMatch')}</p>
  4165. )}
  4166. </div>
  4167. </div>
  4168. <div className="mt-6 flex justify-end gap-3">
  4169. <Button
  4170. variant="secondary"
  4171. onClick={() => {
  4172. setShowChangePasswordModal(false);
  4173. setChangePasswordData({ currentPassword: '', newPassword: '', confirmPassword: '' });
  4174. }}
  4175. >
  4176. Cancel
  4177. </Button>
  4178. <Button
  4179. onClick={async () => {
  4180. if (changePasswordData.newPassword !== changePasswordData.confirmPassword) {
  4181. showToast(t('settings.toast.passwordsDoNotMatch'), 'error');
  4182. return;
  4183. }
  4184. if (changePasswordData.newPassword.length < 6) {
  4185. showToast(t('settings.toast.passwordTooShort'), 'error');
  4186. return;
  4187. }
  4188. setChangePasswordLoading(true);
  4189. try {
  4190. await api.changePassword(changePasswordData.currentPassword, changePasswordData.newPassword);
  4191. showToast(t('settings.toast.passwordChanged'), 'success');
  4192. setShowChangePasswordModal(false);
  4193. setChangePasswordData({ currentPassword: '', newPassword: '', confirmPassword: '' });
  4194. } catch (error: unknown) {
  4195. const message = error instanceof Error ? error.message : 'Failed to change password';
  4196. showToast(message, 'error');
  4197. } finally {
  4198. setChangePasswordLoading(false);
  4199. }
  4200. }}
  4201. disabled={changePasswordLoading || !changePasswordData.currentPassword || !changePasswordData.newPassword || changePasswordData.newPassword !== changePasswordData.confirmPassword || changePasswordData.newPassword.length < 6}
  4202. >
  4203. {changePasswordLoading ? (
  4204. <>
  4205. <Loader2 className="w-4 h-4 animate-spin" />
  4206. Changing...
  4207. </>
  4208. ) : (
  4209. <>
  4210. <Key className="w-4 h-4" />
  4211. Change Password
  4212. </>
  4213. )}
  4214. </Button>
  4215. </div>
  4216. </CardContent>
  4217. </Card>
  4218. </div>
  4219. )}
  4220. </div>
  4221. );
  4222. }