SettingsPage.tsx 203 KB

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