SettingsPage.tsx 214 KB

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