| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347 |
- import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
- 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';
- import { useTranslation } from 'react-i18next';
- import { useNavigate, useSearchParams } from 'react-router-dom';
- import { api } from '../api/client';
- import { useAuth } from '../contexts/AuthContext';
- import { formatDateOnly } from '../utils/date';
- import type { AppSettings, AppSettingsUpdate, SmartPlug, SmartPlugStatus, NotificationProvider, NotificationTemplate, UpdateStatus, GitHubBackupStatus, CloudAuthStatus, UserCreate, UserUpdate, UserResponse, Group, GroupCreate, GroupUpdate, Permission, PermissionCategory } from '../api/client';
- import { Card, CardContent, CardHeader } from '../components/Card';
- import { Button } from '../components/Button';
- import { SmartPlugCard } from '../components/SmartPlugCard';
- import { AddSmartPlugModal } from '../components/AddSmartPlugModal';
- import { NotificationProviderCard } from '../components/NotificationProviderCard';
- import { AddNotificationModal } from '../components/AddNotificationModal';
- import { NotificationTemplateEditor } from '../components/NotificationTemplateEditor';
- import { NotificationLogViewer } from '../components/NotificationLogViewer';
- import { ConfirmModal } from '../components/ConfirmModal';
- import { SpoolmanSettings } from '../components/SpoolmanSettings';
- import { ExternalLinksSettings } from '../components/ExternalLinksSettings';
- import { VirtualPrinterSettings } from '../components/VirtualPrinterSettings';
- import { GitHubBackupSettings } from '../components/GitHubBackupSettings';
- import { APIBrowser } from '../components/APIBrowser';
- import { virtualPrinterApi } from '../api/client';
- import { defaultNavItems, getDefaultView, setDefaultView } from '../components/Layout';
- import { availableLanguages } from '../i18n';
- import { useToast } from '../contexts/ToastContext';
- import { useTheme, type ThemeStyle, type DarkBackground, type LightBackground, type ThemeAccent } from '../contexts/ThemeContext';
- import { useState, useEffect, useRef, useCallback } from 'react';
- import { Palette } from 'lucide-react';
- const validTabs = ['general', 'network', 'plugs', 'notifications', 'filament', 'apikeys', 'virtual-printer', 'users', 'backup'] as const;
- type TabType = typeof validTabs[number];
- export function SettingsPage() {
- const queryClient = useQueryClient();
- const navigate = useNavigate();
- const [searchParams, setSearchParams] = useSearchParams();
- const { t, i18n } = useTranslation();
- const { showToast } = useToast();
- const { authEnabled, user, refreshAuth } = useAuth();
- const {
- mode,
- darkStyle, darkBackground, darkAccent,
- lightStyle, lightBackground, lightAccent,
- setDarkStyle, setDarkBackground, setDarkAccent,
- setLightStyle, setLightBackground, setLightAccent,
- } = useTheme();
- const [localSettings, setLocalSettings] = useState<AppSettings | null>(null);
- const [showPlugModal, setShowPlugModal] = useState(false);
- const [editingPlug, setEditingPlug] = useState<SmartPlug | null>(null);
- const [showNotificationModal, setShowNotificationModal] = useState(false);
- const [editingProvider, setEditingProvider] = useState<NotificationProvider | null>(null);
- const [editingTemplate, setEditingTemplate] = useState<NotificationTemplate | null>(null);
- const [showLogViewer, setShowLogViewer] = useState(false);
- const [defaultView, setDefaultViewState] = useState<string>(getDefaultView());
- // Initialize tab from URL params
- const tabParam = searchParams.get('tab');
- const initialTab = tabParam && validTabs.includes(tabParam as TabType) ? tabParam as TabType : 'general';
- const [activeTab, setActiveTab] = useState<TabType>(initialTab);
- // Update URL when tab changes
- const handleTabChange = (tab: TabType) => {
- setActiveTab(tab);
- if (tab === 'general') {
- searchParams.delete('tab');
- } else {
- searchParams.set('tab', tab);
- }
- setSearchParams(searchParams, { replace: true });
- };
- const [showCreateAPIKey, setShowCreateAPIKey] = useState(false);
- const [newAPIKeyName, setNewAPIKeyName] = useState('');
- const [newAPIKeyPermissions, setNewAPIKeyPermissions] = useState({
- can_queue: true,
- can_control_printer: false,
- can_read_status: true,
- });
- const [createdAPIKey, setCreatedAPIKey] = useState<string | null>(null);
- const [showDeleteAPIKeyConfirm, setShowDeleteAPIKeyConfirm] = useState<number | null>(null);
- const [testApiKey, setTestApiKey] = useState('');
- // Confirm modal states
- const [showClearLogsConfirm, setShowClearLogsConfirm] = useState(false);
- const [showClearStorageConfirm, setShowClearStorageConfirm] = useState(false);
- const [showBulkPlugConfirm, setShowBulkPlugConfirm] = useState<'on' | 'off' | null>(null);
- const [showReleaseNotes, setShowReleaseNotes] = useState(false);
- const [showDisableAuthConfirm, setShowDisableAuthConfirm] = useState(false);
- const [showChangePasswordModal, setShowChangePasswordModal] = useState(false);
- const [changePasswordData, setChangePasswordData] = useState({ currentPassword: '', newPassword: '', confirmPassword: '' });
- const [changePasswordLoading, setChangePasswordLoading] = useState(false);
- // User management state
- const [showCreateUserModal, setShowCreateUserModal] = useState(false);
- const [showEditUserModal, setShowEditUserModal] = useState(false);
- const [editingUserId, setEditingUserId] = useState<number | null>(null);
- const [deleteUserId, setDeleteUserId] = useState<number | null>(null);
- const [deleteUserItemCounts, setDeleteUserItemCounts] = useState<{ archives: number; queue_items: number; library_files: number } | null>(null);
- const [deleteUserLoading, setDeleteUserLoading] = useState(false);
- const [userFormData, setUserFormData] = useState<{
- username: string;
- password: string;
- confirmPassword: string;
- role: string;
- group_ids: number[];
- }>({
- username: '',
- password: '',
- confirmPassword: '',
- role: 'user',
- group_ids: [],
- });
- // Group management state
- const [showCreateGroupModal, setShowCreateGroupModal] = useState(false);
- const [editingGroup, setEditingGroup] = useState<Group | null>(null);
- const [deleteGroupId, setDeleteGroupId] = useState<number | null>(null);
- const [groupFormData, setGroupFormData] = useState<{
- name: string;
- description: string;
- permissions: Permission[];
- }>({
- name: '',
- description: '',
- permissions: [],
- });
- const [expandedCategories, setExpandedCategories] = useState<Set<string>>(new Set());
- // Home Assistant test connection state
- const [haTestResult, setHaTestResult] = useState<{ success: boolean; message: string | null; error: string | null } | null>(null);
- const [haTestLoading, setHaTestLoading] = useState(false);
- // External camera test state
- const [extCameraTestResults, setExtCameraTestResults] = useState<Record<number, { success: boolean; error?: string; resolution?: string } | null>>({});
- const [extCameraTestLoading, setExtCameraTestLoading] = useState<Record<number, boolean>>({});
- const handleDefaultViewChange = (path: string) => {
- setDefaultViewState(path);
- setDefaultView(path);
- showToast(t('settings.toast.settingsSaved'), 'success');
- };
- const handleResetSidebarOrder = () => {
- localStorage.removeItem('sidebarOrder');
- window.location.reload();
- };
- const { data: settings, isLoading } = useQuery({
- queryKey: ['settings'],
- queryFn: api.getSettings,
- });
- const { data: smartPlugs, isLoading: plugsLoading } = useQuery({
- queryKey: ['smart-plugs'],
- queryFn: api.getSmartPlugs,
- });
- // Fetch energy data for all smart plugs when on the plugs tab
- const { data: plugEnergySummary, isLoading: energyLoading } = useQuery({
- queryKey: ['smart-plugs-energy', smartPlugs?.map(p => p.id)],
- queryFn: async () => {
- if (!smartPlugs || smartPlugs.length === 0) return null;
- const statuses = await Promise.all(
- smartPlugs.filter(p => p.enabled).map(async (plug) => {
- try {
- const status = await api.getSmartPlugStatus(plug.id);
- return { plug, status };
- } catch {
- return { plug, status: null as SmartPlugStatus | null };
- }
- })
- );
- // Aggregate energy data
- let totalPower = 0;
- let totalToday = 0;
- let totalYesterday = 0;
- let totalLifetime = 0;
- let reachableCount = 0;
- for (const { plug, status } of statuses) {
- // For MQTT plugs, consider reachable if we have power data
- const hasMqttData = plug.plug_type === 'mqtt' && (status?.energy?.power != null);
- const isReachable = (status?.reachable || hasMqttData) && status?.energy;
- if (isReachable) {
- reachableCount++;
- if (status.energy?.power != null) totalPower += status.energy.power;
- if (status.energy?.today != null) totalToday += status.energy.today;
- if (status.energy?.yesterday != null) totalYesterday += status.energy.yesterday;
- if (status.energy?.total != null) totalLifetime += status.energy.total;
- }
- }
- return {
- totalPower,
- totalToday,
- totalYesterday,
- totalLifetime,
- reachableCount,
- totalPlugs: smartPlugs.filter(p => p.enabled).length,
- };
- },
- enabled: activeTab === 'plugs' && !!smartPlugs && smartPlugs.length > 0,
- refetchInterval: activeTab === 'plugs' ? 10000 : false, // Refresh every 10s when on plugs tab
- });
- const { data: notificationProviders, isLoading: providersLoading } = useQuery({
- queryKey: ['notification-providers'],
- queryFn: api.getNotificationProviders,
- });
- const { data: apiKeys, isLoading: apiKeysLoading } = useQuery({
- queryKey: ['api-keys'],
- queryFn: api.getAPIKeys,
- });
- const createAPIKeyMutation = useMutation({
- mutationFn: (data: { name: string; can_queue: boolean; can_control_printer: boolean; can_read_status: boolean }) =>
- api.createAPIKey(data),
- onSuccess: (data) => {
- setCreatedAPIKey(data.key || null);
- setShowCreateAPIKey(false);
- setNewAPIKeyName('');
- queryClient.invalidateQueries({ queryKey: ['api-keys'] });
- showToast(t('settings.toast.apiKeyCreated'));
- },
- onError: (error: Error) => {
- showToast(`Failed to create API key: ${error.message}`, 'error');
- },
- });
- const deleteAPIKeyMutation = useMutation({
- mutationFn: (id: number) => api.deleteAPIKey(id),
- onSuccess: () => {
- queryClient.invalidateQueries({ queryKey: ['api-keys'] });
- showToast(t('settings.toast.apiKeyDeleted'));
- },
- onError: (error: Error) => {
- showToast(`Failed to delete API key: ${error.message}`, 'error');
- },
- });
- const { data: printers } = useQuery({
- queryKey: ['printers'],
- queryFn: api.getPrinters,
- });
- const { data: notificationTemplates, isLoading: templatesLoading } = useQuery({
- queryKey: ['notification-templates'],
- queryFn: api.getNotificationTemplates,
- });
- // Virtual printer status for tab indicator
- const { data: virtualPrinterSettings } = useQuery({
- queryKey: ['virtual-printer-settings'],
- queryFn: virtualPrinterApi.getSettings,
- refetchInterval: 10000,
- });
- const virtualPrinterRunning = virtualPrinterSettings?.status?.running ?? false;
- const { data: ffmpegStatus } = useQuery({
- queryKey: ['ffmpeg-status'],
- queryFn: api.checkFfmpeg,
- });
- const { data: versionInfo } = useQuery({
- queryKey: ['version'],
- queryFn: api.getVersion,
- });
- const { data: updateCheck, refetch: refetchUpdateCheck, isRefetching: isCheckingUpdate } = useQuery({
- queryKey: ['updateCheck'],
- queryFn: api.checkForUpdates,
- staleTime: 5 * 60 * 1000,
- });
- const { data: updateStatus, refetch: refetchUpdateStatus } = useQuery({
- queryKey: ['updateStatus'],
- queryFn: api.getUpdateStatus,
- refetchInterval: (query) => {
- const status = query.state.data as UpdateStatus | undefined;
- // Poll while update is in progress
- if (status?.status === 'downloading' || status?.status === 'installing') {
- return 1000;
- }
- return false;
- },
- });
- // MQTT status for Network tab
- const { data: mqttStatus } = useQuery({
- queryKey: ['mqtt-status'],
- queryFn: api.getMQTTStatus,
- refetchInterval: activeTab === 'network' ? 5000 : false, // Poll every 5s when on Network tab
- });
- // GitHub backup status for Backup tab indicator
- const { data: githubBackupStatus } = useQuery<GitHubBackupStatus>({
- queryKey: ['github-backup-status'],
- queryFn: api.getGitHubBackupStatus,
- });
- // Cloud auth status for Backup tab indicator
- const { data: cloudAuthStatus } = useQuery<CloudAuthStatus>({
- queryKey: ['cloud-status'],
- queryFn: api.getCloudStatus,
- });
- // User management queries and mutations
- const { hasPermission } = useAuth();
- const { data: usersData = [], isLoading: usersLoading } = useQuery({
- queryKey: ['users'],
- queryFn: () => api.getUsers(),
- enabled: authEnabled && hasPermission('users:read'),
- });
- const { data: groupsData = [], isLoading: groupsLoading } = useQuery({
- queryKey: ['groups'],
- queryFn: () => api.getGroups(),
- enabled: authEnabled && hasPermission('groups:read'),
- });
- const { data: permissionsData } = useQuery({
- queryKey: ['permissions'],
- queryFn: () => api.getPermissions(),
- enabled: authEnabled && hasPermission('groups:read'),
- });
- const createUserMutation = useMutation({
- mutationFn: (data: UserCreate) => api.createUser(data),
- onSuccess: () => {
- queryClient.invalidateQueries({ queryKey: ['users'] });
- queryClient.invalidateQueries({ queryKey: ['groups'] });
- setShowCreateUserModal(false);
- setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
- showToast(t('settings.toast.userCreated'));
- },
- onError: (error: Error) => {
- showToast(error.message, 'error');
- },
- });
- const updateUserMutation = useMutation({
- mutationFn: ({ id, data }: { id: number; data: UserUpdate }) => api.updateUser(id, data),
- onSuccess: () => {
- queryClient.invalidateQueries({ queryKey: ['users'] });
- queryClient.invalidateQueries({ queryKey: ['groups'] });
- setShowEditUserModal(false);
- setEditingUserId(null);
- setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
- showToast(t('settings.toast.userUpdated'));
- },
- onError: (error: Error) => {
- showToast(error.message, 'error');
- },
- });
- const deleteUserMutation = useMutation({
- mutationFn: ({ id, deleteItems }: { id: number; deleteItems: boolean }) => api.deleteUser(id, deleteItems),
- onSuccess: () => {
- queryClient.invalidateQueries({ queryKey: ['users'] });
- showToast(t('settings.toast.userDeleted'));
- setDeleteUserId(null);
- setDeleteUserItemCounts(null);
- },
- onError: (error: Error) => {
- showToast(error.message, 'error');
- },
- });
- // Function to initiate user deletion with item count check
- const handleDeleteUserClick = async (userId: number) => {
- setDeleteUserId(userId);
- setDeleteUserLoading(true);
- try {
- const counts = await api.getUserItemsCount(userId);
- setDeleteUserItemCounts(counts);
- } catch {
- // If we can't get counts, just proceed without showing item options
- setDeleteUserItemCounts({ archives: 0, queue_items: 0, library_files: 0 });
- } finally {
- setDeleteUserLoading(false);
- }
- };
- const createGroupMutation = useMutation({
- mutationFn: (data: GroupCreate) => api.createGroup(data),
- onSuccess: () => {
- queryClient.invalidateQueries({ queryKey: ['groups'] });
- setShowCreateGroupModal(false);
- resetGroupForm();
- showToast(t('settings.toast.groupCreated'));
- },
- onError: (error: Error) => {
- showToast(error.message, 'error');
- },
- });
- const updateGroupMutation = useMutation({
- mutationFn: ({ id, data }: { id: number; data: GroupUpdate }) => api.updateGroup(id, data),
- onSuccess: () => {
- queryClient.invalidateQueries({ queryKey: ['groups'] });
- setEditingGroup(null);
- resetGroupForm();
- showToast(t('settings.toast.groupUpdated'));
- },
- onError: (error: Error) => {
- showToast(error.message, 'error');
- },
- });
- const deleteGroupMutation = useMutation({
- mutationFn: (id: number) => api.deleteGroup(id),
- onSuccess: () => {
- queryClient.invalidateQueries({ queryKey: ['groups'] });
- showToast(t('settings.toast.groupDeleted'));
- },
- onError: (error: Error) => {
- showToast(error.message, 'error');
- },
- });
- // User management handlers
- const handleCreateUser = () => {
- if (!userFormData.username || !userFormData.password) {
- showToast(t('settings.toast.fillRequiredFields'), 'error');
- return;
- }
- if (userFormData.password !== userFormData.confirmPassword) {
- showToast(t('settings.toast.passwordsDoNotMatch'), 'error');
- return;
- }
- if (userFormData.password.length < 6) {
- showToast(t('settings.toast.passwordTooShort'), 'error');
- return;
- }
- createUserMutation.mutate({
- username: userFormData.username,
- password: userFormData.password,
- role: userFormData.role,
- group_ids: userFormData.group_ids.length > 0 ? userFormData.group_ids : undefined,
- });
- };
- const handleUpdateUser = (id: number) => {
- if (userFormData.password) {
- if (userFormData.password !== userFormData.confirmPassword) {
- showToast(t('settings.toast.passwordsDoNotMatch'), 'error');
- return;
- }
- if (userFormData.password.length < 6) {
- showToast(t('settings.toast.passwordTooShort'), 'error');
- return;
- }
- }
- const updateData: UserUpdate = {
- username: userFormData.username || undefined,
- password: userFormData.password || undefined,
- role: userFormData.role,
- group_ids: userFormData.group_ids,
- };
- if (!updateData.password) {
- delete updateData.password;
- }
- updateUserMutation.mutate({ id, data: updateData });
- };
- const startEditUser = (userToEdit: UserResponse) => {
- setEditingUserId(userToEdit.id);
- setUserFormData({
- username: userToEdit.username,
- password: '',
- confirmPassword: '',
- role: userToEdit.role,
- group_ids: userToEdit.groups?.map(g => g.id) || [],
- });
- setShowEditUserModal(true);
- };
- const toggleUserGroup = (groupId: number) => {
- setUserFormData(prev => ({
- ...prev,
- group_ids: prev.group_ids.includes(groupId)
- ? prev.group_ids.filter(id => id !== groupId)
- : [...prev.group_ids, groupId],
- }));
- };
- // Group management handlers
- const resetGroupForm = () => {
- setGroupFormData({ name: '', description: '', permissions: [] });
- setExpandedCategories(new Set());
- };
- const handleCreateGroup = () => {
- if (!groupFormData.name.trim()) {
- showToast(t('settings.toast.enterGroupName'), 'error');
- return;
- }
- createGroupMutation.mutate({
- name: groupFormData.name,
- description: groupFormData.description || undefined,
- permissions: groupFormData.permissions,
- });
- };
- const handleUpdateGroup = () => {
- if (!editingGroup) return;
- if (!groupFormData.name.trim()) {
- showToast(t('settings.toast.enterGroupName'), 'error');
- return;
- }
- updateGroupMutation.mutate({
- id: editingGroup.id,
- data: {
- name: groupFormData.name !== editingGroup.name ? groupFormData.name : undefined,
- description: groupFormData.description,
- permissions: groupFormData.permissions,
- },
- });
- };
- const startEditGroup = (group: Group) => {
- setEditingGroup(group);
- setGroupFormData({
- name: group.name,
- description: group.description || '',
- permissions: group.permissions,
- });
- const cats = new Set<string>();
- permissionsData?.categories.forEach((cat) => {
- if (cat.permissions.some((p) => group.permissions.includes(p.value))) {
- cats.add(cat.name);
- }
- });
- setExpandedCategories(cats);
- };
- const toggleCategory = (categoryName: string) => {
- setExpandedCategories((prev) => {
- const next = new Set(prev);
- if (next.has(categoryName)) {
- next.delete(categoryName);
- } else {
- next.add(categoryName);
- }
- return next;
- });
- };
- const togglePermission = (permission: Permission) => {
- setGroupFormData((prev) => {
- const permissions = prev.permissions.includes(permission)
- ? prev.permissions.filter((p) => p !== permission)
- : [...prev.permissions, permission];
- return { ...prev, permissions };
- });
- };
- const toggleCategoryPermissions = (category: PermissionCategory, checked: boolean) => {
- setGroupFormData((prev) => {
- const categoryPerms = category.permissions.map((p) => p.value);
- const otherPerms = prev.permissions.filter((p) => !categoryPerms.includes(p));
- const permissions = checked ? [...otherPerms, ...categoryPerms] : otherPerms;
- return { ...prev, permissions };
- });
- };
- const isCategoryFullySelected = (category: PermissionCategory) => {
- return category.permissions.every((p) => groupFormData.permissions.includes(p.value));
- };
- const isCategoryPartiallySelected = (category: PermissionCategory) => {
- const selected = category.permissions.filter((p) => groupFormData.permissions.includes(p.value));
- return selected.length > 0 && selected.length < category.permissions.length;
- };
- const applyUpdateMutation = useMutation({
- mutationFn: api.applyUpdate,
- onSuccess: (data) => {
- if (data.is_docker) {
- showToast(data.message, 'error');
- } else {
- refetchUpdateStatus();
- }
- },
- });
- // Test all notification providers
- const [testAllResult, setTestAllResult] = useState<{
- tested: number;
- success: number;
- failed: number;
- results: Array<{
- provider_id: number;
- provider_name: string;
- provider_type: string;
- success: boolean;
- message: string;
- }>;
- } | null>(null);
- const testAllMutation = useMutation({
- mutationFn: api.testAllNotificationProviders,
- onSuccess: (data) => {
- setTestAllResult(data);
- queryClient.invalidateQueries({ queryKey: ['notification-providers'] });
- if (data.failed === 0) {
- showToast(`All ${data.tested} providers tested successfully!`, 'success');
- } else {
- showToast(`${data.success}/${data.tested} providers succeeded`, data.failed > 0 ? 'error' : 'success');
- }
- },
- onError: (error: Error) => {
- showToast(`Failed to test providers: ${error.message}`, 'error');
- },
- });
- // Bulk action for smart plugs
- const bulkPlugActionMutation = useMutation({
- mutationFn: async (action: 'on' | 'off') => {
- if (!smartPlugs) return { success: 0, failed: 0 };
- const enabledPlugs = smartPlugs.filter(p => p.enabled);
- const results = await Promise.all(
- enabledPlugs.map(async (plug) => {
- try {
- await api.controlSmartPlug(plug.id, action);
- return { success: true };
- } catch {
- return { success: false };
- }
- })
- );
- return {
- success: results.filter(r => r.success).length,
- failed: results.filter(r => !r.success).length,
- };
- },
- onSuccess: (data, action) => {
- queryClient.invalidateQueries({ queryKey: ['smart-plugs'] });
- queryClient.invalidateQueries({ queryKey: ['smart-plugs-energy'] });
- if (data.failed === 0) {
- showToast(`All ${data.success} plugs turned ${action}`, 'success');
- } else {
- showToast(`${data.success} plugs turned ${action}, ${data.failed} failed`, 'error');
- }
- },
- onError: (error: Error) => {
- showToast(`Failed: ${error.message}`, 'error');
- },
- });
- // Ref for debounce timeout
- const saveTimeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null);
- const isSavingRef = useRef(false);
- const isInitialLoadRef = useRef(true);
- // Sync local state when settings load
- useEffect(() => {
- if (settings && !localSettings) {
- // Auto-detect external_url from browser if not set
- const settingsWithExternalUrl = {
- ...settings,
- external_url: settings.external_url || window.location.origin,
- };
- setLocalSettings(settingsWithExternalUrl);
- // Mark initial load complete after a short delay
- setTimeout(() => {
- isInitialLoadRef.current = false;
- }, 100);
- }
- }, [settings, localSettings]);
- const updateMutation = useMutation({
- mutationFn: api.updateSettings,
- onSuccess: (data) => {
- queryClient.setQueryData(['settings'], data);
- // Sync localSettings with the saved data to prevent re-triggering saves
- setLocalSettings(data);
- // Invalidate archive stats to reflect energy tracking mode change
- queryClient.invalidateQueries({ queryKey: ['archiveStats'] });
- showToast(t('settings.toast.settingsSaved'), 'success');
- },
- onError: (error: Error) => {
- showToast(`Failed to save: ${error.message}`, 'error');
- },
- onSettled: () => {
- // Reset saving flag when mutation completes (success or error)
- isSavingRef.current = false;
- },
- });
- const updatePrinterMutation = useMutation({
- mutationFn: ({ id, data }: { id: number; data: Partial<{ external_camera_url: string | null; external_camera_type: string | null; external_camera_enabled: boolean }> }) =>
- api.updatePrinter(id, data),
- onSuccess: () => {
- queryClient.invalidateQueries({ queryKey: ['printers'] });
- showToast(t('settings.toast.cameraSettingsSaved'), 'success');
- },
- onError: (error: Error) => {
- showToast(`Failed to update printer: ${error.message}`, 'error');
- },
- });
- // Debounced auto-save when localSettings change
- useEffect(() => {
- // Skip if initial load or no settings
- if (isInitialLoadRef.current || !localSettings || !settings) {
- return;
- }
- // Check if there are actual changes
- const hasChanges =
- settings.auto_archive !== localSettings.auto_archive ||
- settings.save_thumbnails !== localSettings.save_thumbnails ||
- settings.capture_finish_photo !== localSettings.capture_finish_photo ||
- settings.default_filament_cost !== localSettings.default_filament_cost ||
- settings.currency !== localSettings.currency ||
- settings.energy_cost_per_kwh !== localSettings.energy_cost_per_kwh ||
- settings.energy_tracking_mode !== localSettings.energy_tracking_mode ||
- settings.check_updates !== localSettings.check_updates ||
- (settings.check_printer_firmware ?? true) !== (localSettings.check_printer_firmware ?? true) ||
- settings.notification_language !== localSettings.notification_language ||
- settings.ams_humidity_good !== localSettings.ams_humidity_good ||
- settings.ams_humidity_fair !== localSettings.ams_humidity_fair ||
- settings.ams_temp_good !== localSettings.ams_temp_good ||
- settings.ams_temp_fair !== localSettings.ams_temp_fair ||
- settings.ams_history_retention_days !== localSettings.ams_history_retention_days ||
- settings.per_printer_mapping_expanded !== localSettings.per_printer_mapping_expanded ||
- settings.date_format !== localSettings.date_format ||
- settings.time_format !== localSettings.time_format ||
- settings.default_printer_id !== localSettings.default_printer_id ||
- settings.ftp_retry_enabled !== localSettings.ftp_retry_enabled ||
- settings.ftp_retry_count !== localSettings.ftp_retry_count ||
- settings.ftp_retry_delay !== localSettings.ftp_retry_delay ||
- settings.ftp_timeout !== localSettings.ftp_timeout ||
- settings.mqtt_enabled !== localSettings.mqtt_enabled ||
- settings.mqtt_broker !== localSettings.mqtt_broker ||
- settings.mqtt_port !== localSettings.mqtt_port ||
- settings.mqtt_username !== localSettings.mqtt_username ||
- settings.mqtt_password !== localSettings.mqtt_password ||
- settings.mqtt_topic_prefix !== localSettings.mqtt_topic_prefix ||
- settings.mqtt_use_tls !== localSettings.mqtt_use_tls ||
- settings.external_url !== localSettings.external_url ||
- settings.ha_enabled !== localSettings.ha_enabled ||
- settings.ha_url !== localSettings.ha_url ||
- settings.ha_token !== localSettings.ha_token ||
- (settings.library_archive_mode ?? 'ask') !== (localSettings.library_archive_mode ?? 'ask') ||
- Number(settings.library_disk_warning_gb ?? 5) !== Number(localSettings.library_disk_warning_gb ?? 5) ||
- (settings.camera_view_mode ?? 'window') !== (localSettings.camera_view_mode ?? 'window') ||
- settings.prometheus_enabled !== localSettings.prometheus_enabled ||
- settings.prometheus_token !== localSettings.prometheus_token;
- if (!hasChanges) {
- return;
- }
- // Don't queue more saves while one is in progress
- if (isSavingRef.current) {
- return;
- }
- // Clear existing timeout
- if (saveTimeoutRef.current) {
- clearTimeout(saveTimeoutRef.current);
- }
- // Set new debounced save (500ms delay)
- saveTimeoutRef.current = setTimeout(() => {
- // Skip if a save is already in progress
- if (isSavingRef.current) {
- return;
- }
- isSavingRef.current = true;
- // Only send the fields we manage on this page (exclude virtual_printer_* which are managed separately)
- const settingsToSave: AppSettingsUpdate = {
- auto_archive: localSettings.auto_archive,
- save_thumbnails: localSettings.save_thumbnails,
- capture_finish_photo: localSettings.capture_finish_photo,
- default_filament_cost: localSettings.default_filament_cost,
- currency: localSettings.currency,
- energy_cost_per_kwh: localSettings.energy_cost_per_kwh,
- energy_tracking_mode: localSettings.energy_tracking_mode,
- check_updates: localSettings.check_updates,
- check_printer_firmware: localSettings.check_printer_firmware,
- notification_language: localSettings.notification_language,
- ams_humidity_good: localSettings.ams_humidity_good,
- ams_humidity_fair: localSettings.ams_humidity_fair,
- ams_temp_good: localSettings.ams_temp_good,
- ams_temp_fair: localSettings.ams_temp_fair,
- ams_history_retention_days: localSettings.ams_history_retention_days,
- per_printer_mapping_expanded: localSettings.per_printer_mapping_expanded,
- date_format: localSettings.date_format,
- time_format: localSettings.time_format,
- default_printer_id: localSettings.default_printer_id,
- ftp_retry_enabled: localSettings.ftp_retry_enabled,
- ftp_retry_count: localSettings.ftp_retry_count,
- ftp_retry_delay: localSettings.ftp_retry_delay,
- ftp_timeout: localSettings.ftp_timeout,
- mqtt_enabled: localSettings.mqtt_enabled,
- mqtt_broker: localSettings.mqtt_broker,
- mqtt_port: localSettings.mqtt_port,
- mqtt_username: localSettings.mqtt_username,
- mqtt_password: localSettings.mqtt_password,
- mqtt_topic_prefix: localSettings.mqtt_topic_prefix,
- mqtt_use_tls: localSettings.mqtt_use_tls,
- external_url: localSettings.external_url,
- ha_enabled: localSettings.ha_enabled,
- ha_url: localSettings.ha_url,
- ha_token: localSettings.ha_token,
- library_archive_mode: localSettings.library_archive_mode,
- library_disk_warning_gb: localSettings.library_disk_warning_gb,
- camera_view_mode: localSettings.camera_view_mode,
- prometheus_enabled: localSettings.prometheus_enabled,
- prometheus_token: localSettings.prometheus_token,
- };
- updateMutation.mutate(settingsToSave);
- }, 500);
- // Cleanup on unmount or when localSettings changes again
- return () => {
- if (saveTimeoutRef.current) {
- clearTimeout(saveTimeoutRef.current);
- }
- };
- }, [localSettings, settings, updateMutation]);
- const updateSetting = useCallback(<K extends keyof AppSettings>(key: K, value: AppSettings[K]) => {
- setLocalSettings(prev => prev ? { ...prev, [key]: value } : null);
- }, []);
- const handleTestExternalCamera = async (printerId: number, url: string, cameraType: string) => {
- if (!url) {
- showToast(t('settings.toast.enterCameraUrl'), 'error');
- return;
- }
- setExtCameraTestLoading(prev => ({ ...prev, [printerId]: true }));
- setExtCameraTestResults(prev => ({ ...prev, [printerId]: null }));
- try {
- const result = await api.testExternalCamera(printerId, url, cameraType);
- setExtCameraTestResults(prev => ({ ...prev, [printerId]: result }));
- if (result.success) {
- showToast(t('settings.toast.cameraConnected', { resolution: result.resolution || '' }), 'success');
- } else {
- showToast(result.error || t('settings.toast.connectionFailed'), 'error');
- }
- } catch (error) {
- const message = error instanceof Error ? error.message : t('settings.toast.testFailed');
- setExtCameraTestResults(prev => ({ ...prev, [printerId]: { success: false, error: message } }));
- showToast(message, 'error');
- } finally {
- setExtCameraTestLoading(prev => ({ ...prev, [printerId]: false }));
- }
- };
- // Local state for camera URL inputs (to avoid saving on every keystroke)
- const [localCameraUrls, setLocalCameraUrls] = useState<Record<number, string>>({});
- const cameraUrlSaveTimeoutRef = useRef<Record<number, ReturnType<typeof setTimeout>>>({});
- const initializedPrinterUrlsRef = useRef<Set<number>>(new Set());
- // Initialize local camera URLs from printer data
- useEffect(() => {
- if (printers) {
- const urls: Record<number, string> = {};
- printers.forEach(p => {
- if (p.external_camera_url && !initializedPrinterUrlsRef.current.has(p.id)) {
- urls[p.id] = p.external_camera_url;
- initializedPrinterUrlsRef.current.add(p.id);
- }
- });
- if (Object.keys(urls).length > 0) {
- setLocalCameraUrls(prev => ({ ...prev, ...urls }));
- }
- }
- }, [printers]);
- const handleCameraUrlChange = (printerId: number, url: string) => {
- // Update local state immediately for responsive UI
- setLocalCameraUrls(prev => ({ ...prev, [printerId]: url }));
- // Clear existing timeout for this printer
- if (cameraUrlSaveTimeoutRef.current[printerId]) {
- clearTimeout(cameraUrlSaveTimeoutRef.current[printerId]);
- }
- // Debounce the save (800ms delay)
- cameraUrlSaveTimeoutRef.current[printerId] = setTimeout(() => {
- updatePrinterMutation.mutate({
- id: printerId,
- data: { external_camera_url: url || null }
- });
- }, 800);
- };
- const handleUpdatePrinterCamera = (printerId: number, updates: { type?: string; enabled?: boolean }) => {
- const data: Partial<{ external_camera_type: string | null; external_camera_enabled: boolean }> = {};
- if (updates.type !== undefined) data.external_camera_type = updates.type || null;
- if (updates.enabled !== undefined) data.external_camera_enabled = updates.enabled;
- updatePrinterMutation.mutate({ id: printerId, data });
- };
- if (isLoading || !localSettings) {
- return (
- <div className="p-4 md:p-8 flex justify-center">
- <Loader2 className="w-8 h-8 text-bambu-green animate-spin" />
- </div>
- );
- }
- return (
- <div className="p-4 md:p-8">
- <div className="mb-8">
- <h1 className="text-2xl font-bold text-white">{t('settings.title')}</h1>
- <p className="text-bambu-gray">{t('settings.configureBambuddy')}</p>
- </div>
- {/* Tab Navigation */}
- <div className="flex gap-1 mb-6 border-b border-bambu-dark-tertiary overflow-x-auto">
- <button
- onClick={() => handleTabChange('general')}
- className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px ${
- activeTab === 'general'
- ? 'text-bambu-green border-bambu-green'
- : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
- }`}
- >
- {t('settings.tabs.general')}
- </button>
- <button
- onClick={() => handleTabChange('plugs')}
- className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px flex items-center gap-2 ${
- activeTab === 'plugs'
- ? 'text-bambu-green border-bambu-green'
- : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
- }`}
- >
- <Plug className="w-4 h-4" />
- {t('settings.tabs.smartPlugs')}
- {smartPlugs && smartPlugs.length > 0 && (
- <span className="text-xs bg-bambu-dark-tertiary px-1.5 py-0.5 rounded-full">
- {smartPlugs.length}
- </span>
- )}
- </button>
- <button
- onClick={() => handleTabChange('notifications')}
- className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px flex items-center gap-2 ${
- activeTab === 'notifications'
- ? 'text-bambu-green border-bambu-green'
- : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
- }`}
- >
- <Bell className="w-4 h-4" />
- {t('settings.tabs.notifications')}
- {notificationProviders && notificationProviders.length > 0 && (
- <span className="text-xs bg-bambu-dark-tertiary px-1.5 py-0.5 rounded-full">
- {notificationProviders.length}
- </span>
- )}
- </button>
- <button
- onClick={() => handleTabChange('filament')}
- className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px flex items-center gap-2 ${
- activeTab === 'filament'
- ? 'text-bambu-green border-bambu-green'
- : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
- }`}
- >
- <Cylinder className="w-4 h-4" />
- {t('settings.tabs.filament')}
- </button>
- <button
- onClick={() => handleTabChange('network')}
- className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px flex items-center gap-2 ${
- activeTab === 'network'
- ? 'text-bambu-green border-bambu-green'
- : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
- }`}
- >
- <Wifi className="w-4 h-4" />
- {t('settings.tabs.network')}
- <span className={`w-2 h-2 rounded-full ${mqttStatus?.enabled ? 'bg-green-400' : 'bg-gray-500'}`} />
- </button>
- <button
- onClick={() => handleTabChange('apikeys')}
- className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px flex items-center gap-2 ${
- activeTab === 'apikeys'
- ? 'text-bambu-green border-bambu-green'
- : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
- }`}
- >
- <Key className="w-4 h-4" />
- {t('settings.tabs.apiKeys')}
- {apiKeys && apiKeys.length > 0 && (
- <span className="text-xs bg-bambu-dark-tertiary px-1.5 py-0.5 rounded-full">
- {apiKeys.length}
- </span>
- )}
- </button>
- <button
- onClick={() => handleTabChange('virtual-printer')}
- className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px flex items-center gap-2 ${
- activeTab === 'virtual-printer'
- ? 'text-bambu-green border-bambu-green'
- : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
- }`}
- >
- <Printer className="w-4 h-4" />
- {t('settings.tabs.virtualPrinter')}
- <span className={`w-2 h-2 rounded-full ${virtualPrinterRunning ? 'bg-green-400' : 'bg-gray-500'}`} />
- </button>
- <button
- onClick={() => handleTabChange('users')}
- className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px flex items-center gap-2 ${
- activeTab === 'users'
- ? 'text-bambu-green border-bambu-green'
- : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
- }`}
- >
- <Users className="w-4 h-4" />
- {t('settings.tabs.users')}
- {authEnabled && (
- <span className="w-2 h-2 rounded-full bg-green-400" />
- )}
- </button>
- <button
- onClick={() => handleTabChange('backup')}
- className={`px-4 py-2 text-sm font-medium transition-colors border-b-2 -mb-px flex items-center gap-2 ${
- activeTab === 'backup'
- ? 'text-bambu-green border-bambu-green'
- : 'text-bambu-gray hover:text-gray-900 dark:hover:text-white border-transparent'
- }`}
- >
- <Database className="w-4 h-4" />
- {t('settings.tabs.backup')}
- <span className={`w-2 h-2 rounded-full ${cloudAuthStatus?.is_authenticated && githubBackupStatus?.configured && githubBackupStatus?.enabled ? 'bg-green-400' : 'bg-gray-500'}`} />
- </button>
- </div>
- {/* General Tab */}
- {activeTab === 'general' && (
- <div className="flex flex-col lg:flex-row gap-6 lg:gap-8">
- {/* Left Column - General Settings */}
- <div className="space-y-6 flex-1 lg:max-w-xl">
- <Card>
- <CardHeader>
- <h2 className="text-lg font-semibold text-white">{t('settings.general')}</h2>
- </CardHeader>
- <CardContent className="space-y-4">
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- <Globe className="w-4 h-4 inline mr-1" />
- {t('settings.language')}
- </label>
- <div className="relative">
- <select
- value={i18n.language}
- onChange={(e) => { i18n.changeLanguage(e.target.value); showToast(t('settings.toast.settingsSaved'), 'success'); }}
- 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"
- >
- {availableLanguages.map((lang) => (
- <option key={lang.code} value={lang.code}>
- {lang.nativeName} ({lang.name})
- </option>
- ))}
- </select>
- <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
- </div>
- <p className="text-xs text-bambu-gray mt-1">
- {t('settings.languageDescription')}
- </p>
- </div>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- {t('settings.defaultView')}
- </label>
- <div className="relative">
- <select
- value={defaultView}
- onChange={(e) => handleDefaultViewChange(e.target.value)}
- 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"
- >
- {defaultNavItems.map((item) => (
- <option key={item.id} value={item.to}>
- {t(item.labelKey)}
- </option>
- ))}
- </select>
- <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
- </div>
- <p className="text-xs text-bambu-gray mt-1">
- {t('settings.defaultViewDescription')}
- </p>
- </div>
- <div className="grid grid-cols-2 gap-3">
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Date Format
- </label>
- <div className="relative">
- <select
- value={localSettings.date_format || 'system'}
- onChange={(e) => updateSetting('date_format', e.target.value as 'system' | 'us' | 'eu' | 'iso')}
- 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"
- >
- <option value="system">{t('settings.systemDefault')}</option>
- <option value="us">US (MM/DD/YYYY)</option>
- <option value="eu">EU (DD/MM/YYYY)</option>
- <option value="iso">ISO (YYYY-MM-DD)</option>
- </select>
- <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
- </div>
- </div>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Time Format
- </label>
- <div className="relative">
- <select
- value={localSettings.time_format || 'system'}
- onChange={(e) => updateSetting('time_format', e.target.value as 'system' | '12h' | '24h')}
- 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"
- >
- <option value="system">{t('settings.systemDefault')}</option>
- <option value="12h">12-hour (3:30 PM)</option>
- <option value="24h">24-hour (15:30)</option>
- </select>
- <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
- </div>
- </div>
- </div>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Default Printer
- </label>
- <div className="relative">
- <select
- value={localSettings.default_printer_id ?? ''}
- onChange={(e) => updateSetting('default_printer_id', e.target.value ? Number(e.target.value) : null)}
- 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"
- >
- <option value="">{t('settings.noDefaultPrinter')}</option>
- {printers?.map((printer) => (
- <option key={printer.id} value={printer.id}>
- {printer.name}
- </option>
- ))}
- </select>
- <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
- </div>
- <p className="text-xs text-bambu-gray mt-1">
- Pre-select this printer for uploads, reprints, and other operations.
- </p>
- </div>
- <div className="flex items-center justify-between">
- <div>
- <p className="text-white">{t('settings.sidebarOrder')}</p>
- <p className="text-sm text-bambu-gray">
- Drag items in the sidebar to reorder. Reset to default order here.
- </p>
- </div>
- <Button
- variant="secondary"
- size="sm"
- onClick={handleResetSidebarOrder}
- >
- <RotateCcw className="w-4 h-4" />
- Reset
- </Button>
- </div>
- </CardContent>
- </Card>
- <Card>
- <CardHeader>
- <h2 className="text-lg font-semibold text-white flex items-center gap-2">
- <Palette className="w-5 h-5" />
- Appearance
- </h2>
- </CardHeader>
- <CardContent className="space-y-6">
- {/* Dark Mode Settings */}
- <div className={`space-y-3 p-4 rounded-lg border ${mode === 'dark' ? 'border-bambu-green bg-bambu-green/5' : 'border-bambu-dark-tertiary'}`}>
- <h3 className="text-sm font-medium text-white flex items-center gap-2">
- Dark Mode
- {mode === 'dark' && <span className="text-xs text-bambu-green">(active)</span>}
- </h3>
- <div className="grid grid-cols-3 gap-3">
- <div>
- <label className="block text-xs text-bambu-gray mb-1">Background</label>
- <select
- value={darkBackground}
- onChange={(e) => { setDarkBackground(e.target.value as DarkBackground); showToast(t('settings.toast.settingsSaved'), 'success'); }}
- 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"
- >
- <option value="neutral">Neutral</option>
- <option value="warm">Warm</option>
- <option value="cool">Cool</option>
- <option value="oled">OLED Black</option>
- <option value="slate">Slate Blue</option>
- <option value="forest">Forest Green</option>
- </select>
- </div>
- <div>
- <label className="block text-xs text-bambu-gray mb-1">Accent</label>
- <select
- value={darkAccent}
- onChange={(e) => { setDarkAccent(e.target.value as ThemeAccent); showToast(t('settings.toast.settingsSaved'), 'success'); }}
- 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"
- >
- <option value="green">Green</option>
- <option value="teal">Teal</option>
- <option value="blue">Blue</option>
- <option value="orange">Orange</option>
- <option value="purple">Purple</option>
- <option value="red">Red</option>
- </select>
- </div>
- <div>
- <label className="block text-xs text-bambu-gray mb-1">Style</label>
- <select
- value={darkStyle}
- onChange={(e) => { setDarkStyle(e.target.value as ThemeStyle); showToast(t('settings.toast.settingsSaved'), 'success'); }}
- 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"
- >
- <option value="classic">Classic</option>
- <option value="glow">Glow</option>
- <option value="vibrant">Vibrant</option>
- </select>
- </div>
- </div>
- </div>
- {/* Light Mode Settings */}
- <div className={`space-y-3 p-4 rounded-lg border ${mode === 'light' ? 'border-bambu-green bg-bambu-green/5' : 'border-bambu-dark-tertiary'}`}>
- <h3 className="text-sm font-medium text-white flex items-center gap-2">
- Light Mode
- {mode === 'light' && <span className="text-xs text-bambu-green">(active)</span>}
- </h3>
- <div className="grid grid-cols-3 gap-3">
- <div>
- <label className="block text-xs text-bambu-gray mb-1">Background</label>
- <select
- value={lightBackground}
- onChange={(e) => { setLightBackground(e.target.value as LightBackground); showToast(t('settings.toast.settingsSaved'), 'success'); }}
- 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"
- >
- <option value="neutral">Neutral</option>
- <option value="warm">Warm</option>
- <option value="cool">Cool</option>
- </select>
- </div>
- <div>
- <label className="block text-xs text-bambu-gray mb-1">Accent</label>
- <select
- value={lightAccent}
- onChange={(e) => { setLightAccent(e.target.value as ThemeAccent); showToast(t('settings.toast.settingsSaved'), 'success'); }}
- 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"
- >
- <option value="green">Green</option>
- <option value="teal">Teal</option>
- <option value="blue">Blue</option>
- <option value="orange">Orange</option>
- <option value="purple">Purple</option>
- <option value="red">Red</option>
- </select>
- </div>
- <div>
- <label className="block text-xs text-bambu-gray mb-1">Style</label>
- <select
- value={lightStyle}
- onChange={(e) => { setLightStyle(e.target.value as ThemeStyle); showToast(t('settings.toast.settingsSaved'), 'success'); }}
- 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"
- >
- <option value="classic">Classic</option>
- <option value="glow">Glow</option>
- <option value="vibrant">Vibrant</option>
- </select>
- </div>
- </div>
- </div>
- <p className="text-xs text-bambu-gray">
- Toggle between dark and light mode using the sun/moon icon in the sidebar.
- </p>
- </CardContent>
- </Card>
- <Card>
- <CardHeader>
- <h2 className="text-lg font-semibold text-white">{t('settings.archiveSettings')}</h2>
- </CardHeader>
- <CardContent className="space-y-4">
- <div className="flex items-center justify-between">
- <div>
- <p className="text-white">Auto-archive prints</p>
- <p className="text-sm text-bambu-gray">
- Automatically save 3MF files when prints complete
- </p>
- </div>
- <label className="relative inline-flex items-center cursor-pointer">
- <input
- type="checkbox"
- checked={localSettings.auto_archive}
- onChange={(e) => updateSetting('auto_archive', e.target.checked)}
- className="sr-only peer"
- />
- <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>
- </label>
- </div>
- <div className="flex items-center justify-between">
- <div>
- <p className="text-white">{t('settings.saveThumbnails')}</p>
- <p className="text-sm text-bambu-gray">
- Extract and save preview images from 3MF files
- </p>
- </div>
- <label className="relative inline-flex items-center cursor-pointer">
- <input
- type="checkbox"
- checked={localSettings.save_thumbnails}
- onChange={(e) => updateSetting('save_thumbnails', e.target.checked)}
- className="sr-only peer"
- />
- <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>
- </label>
- </div>
- <div className="flex items-center justify-between">
- <div>
- <p className="text-white">{t('settings.captureFinishPhoto')}</p>
- <p className="text-sm text-bambu-gray">
- Take a photo from printer camera when print completes
- </p>
- </div>
- <label className="relative inline-flex items-center cursor-pointer">
- <input
- type="checkbox"
- checked={localSettings.capture_finish_photo}
- onChange={(e) => updateSetting('capture_finish_photo', e.target.checked)}
- className="sr-only peer"
- />
- <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>
- </label>
- </div>
- {localSettings.capture_finish_photo && ffmpegStatus && !ffmpegStatus.installed && (
- <div className="flex items-start gap-2 p-3 bg-yellow-500/10 border border-yellow-500/30 rounded-lg">
- <AlertTriangle className="w-5 h-5 text-yellow-500 flex-shrink-0 mt-0.5" />
- <div className="text-sm">
- <p className="text-yellow-500 font-medium">ffmpeg not installed</p>
- <p className="text-bambu-gray mt-1">
- Camera capture requires ffmpeg. Install it via{' '}
- <code className="bg-bambu-dark-tertiary px-1 rounded">brew install ffmpeg</code> (macOS) or{' '}
- <code className="bg-bambu-dark-tertiary px-1 rounded">apt install ffmpeg</code> (Linux).
- </p>
- </div>
- </div>
- )}
- </CardContent>
- </Card>
- </div>
- {/* Second Column - Camera, Cost, AMS & Spoolman */}
- <div className="space-y-6 flex-1 lg:max-w-md">
- {/* Camera Settings */}
- <Card>
- <CardHeader>
- <h2 className="text-lg font-semibold text-white flex items-center gap-2">
- <Video className="w-5 h-5 text-bambu-green" />
- Camera
- </h2>
- </CardHeader>
- <CardContent className="space-y-4">
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Camera View Mode
- </label>
- <select
- value={localSettings.camera_view_mode ?? 'window'}
- onChange={(e) => updateSetting('camera_view_mode', e.target.value as 'window' | 'embedded')}
- 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"
- >
- <option value="window">{t('settings.newWindow')}</option>
- <option value="embedded">{t('settings.embeddedOverlay')}</option>
- </select>
- <p className="text-xs text-bambu-gray mt-1">
- {localSettings.camera_view_mode === 'embedded'
- ? 'Camera opens in a resizable overlay on the main screen'
- : 'Camera opens in a separate browser window'}
- </p>
- </div>
- {/* External Cameras Section */}
- <div className="border-t border-bambu-dark-tertiary pt-4 mt-4">
- <h3 className="text-sm font-medium text-white mb-2">{t('settings.externalCameras')}</h3>
- <p className="text-xs text-bambu-gray mb-3">
- 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.
- </p>
- {printers && printers.length > 0 ? (
- <div className="space-y-3">
- {printers.map(printer => (
- <div key={printer.id} className="p-3 bg-bambu-dark rounded-lg">
- <div className="flex items-center justify-between mb-2">
- <span className="text-white font-medium text-sm">{printer.name}</span>
- <label className="relative inline-flex items-center cursor-pointer">
- <input
- type="checkbox"
- checked={printer.external_camera_enabled}
- onChange={(e) => handleUpdatePrinterCamera(printer.id, { enabled: e.target.checked })}
- className="sr-only peer"
- />
- <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>
- </label>
- </div>
- {printer.external_camera_enabled && (
- <div className="space-y-2 mt-2">
- <input
- type="text"
- placeholder={printer.external_camera_type === 'usb' ? 'Device path (/dev/video0)' : 'Camera URL (rtsp://... or http://...)'}
- value={localCameraUrls[printer.id] ?? printer.external_camera_url ?? ''}
- onChange={(e) => handleCameraUrlChange(printer.id, e.target.value)}
- 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"
- />
- <div className="flex gap-2">
- <select
- value={printer.external_camera_type || 'mjpeg'}
- onChange={(e) => handleUpdatePrinterCamera(printer.id, { type: e.target.value })}
- 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"
- >
- <option value="mjpeg">MJPEG Stream</option>
- <option value="rtsp">RTSP Stream</option>
- <option value="snapshot">HTTP Snapshot</option>
- <option value="usb">USB Camera (V4L2)</option>
- </select>
- <Button
- size="sm"
- variant="secondary"
- onClick={() => handleTestExternalCamera(printer.id, localCameraUrls[printer.id] ?? printer.external_camera_url ?? '', printer.external_camera_type || 'mjpeg')}
- disabled={extCameraTestLoading[printer.id] || !(localCameraUrls[printer.id] ?? printer.external_camera_url)}
- >
- {extCameraTestLoading[printer.id] ? (
- <Loader2 className="w-4 h-4 animate-spin" />
- ) : (
- 'Test'
- )}
- </Button>
- </div>
- {extCameraTestResults[printer.id] && (
- <div className={`text-xs flex items-center gap-1 ${extCameraTestResults[printer.id]?.success ? 'text-green-500' : 'text-red-500'}`}>
- {extCameraTestResults[printer.id]?.success ? (
- <>
- <CheckCircle className="w-3 h-3" />
- Connected{extCameraTestResults[printer.id]?.resolution && ` (${extCameraTestResults[printer.id]?.resolution})`}
- </>
- ) : (
- <>
- <XCircle className="w-3 h-3" />
- {extCameraTestResults[printer.id]?.error || t('settings.toast.connectionFailed')}
- </>
- )}
- </div>
- )}
- </div>
- )}
- </div>
- ))}
- </div>
- ) : (
- <p className="text-xs text-bambu-gray italic">{t('settings.noPrintersConfigured')}</p>
- )}
- </div>
- </CardContent>
- </Card>
- <Card>
- <CardHeader>
- <h2 className="text-lg font-semibold text-white">{t('settings.costTracking')}</h2>
- </CardHeader>
- <CardContent className="space-y-4">
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Default filament cost (per kg)
- </label>
- <input
- type="number"
- step="0.01"
- min="0"
- value={localSettings.default_filament_cost}
- onChange={(e) =>
- updateSetting('default_filament_cost', parseFloat(e.target.value) || 0)
- }
- 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"
- />
- </div>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">Currency</label>
- <select
- value={localSettings.currency}
- onChange={(e) => updateSetting('currency', e.target.value)}
- 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"
- >
- <option value="USD">USD ($)</option>
- <option value="EUR">EUR (€)</option>
- <option value="GBP">GBP (£)</option>
- <option value="CHF">CHF (Fr.)</option>
- <option value="JPY">JPY (¥)</option>
- <option value="CNY">CNY (¥)</option>
- <option value="CAD">CAD ($)</option>
- <option value="AUD">AUD ($)</option>
- </select>
- </div>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Electricity cost per kWh
- </label>
- <input
- type="number"
- step="0.01"
- min="0"
- value={localSettings.energy_cost_per_kwh}
- onChange={(e) =>
- updateSetting('energy_cost_per_kwh', parseFloat(e.target.value) || 0)
- }
- 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"
- />
- </div>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Energy display mode
- </label>
- <select
- value={localSettings.energy_tracking_mode || 'total'}
- onChange={(e) => updateSetting('energy_tracking_mode', e.target.value as 'print' | 'total')}
- 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"
- >
- <option value="print">{t('settings.printsOnly')}</option>
- <option value="total">{t('settings.totalConsumption')}</option>
- </select>
- <p className="text-xs text-bambu-gray mt-1">
- {localSettings.energy_tracking_mode === 'print'
- ? 'Dashboard shows sum of energy used during prints'
- : 'Dashboard shows lifetime energy from smart plugs'}
- </p>
- </div>
- </CardContent>
- </Card>
- {/* File Manager Settings */}
- <Card>
- <CardHeader>
- <h2 className="text-lg font-semibold text-white flex items-center gap-2">
- <FileText className="w-5 h-5 text-bambu-green" />
- File Manager
- </h2>
- </CardHeader>
- <CardContent className="space-y-4">
- {/* Archive Mode */}
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Create Archive Entry When Printing
- </label>
- <select
- value={localSettings.library_archive_mode ?? 'ask'}
- onChange={(e) => updateSetting('library_archive_mode', e.target.value as 'always' | 'never' | 'ask')}
- 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"
- >
- <option value="always">{t('settings.archiveMode.always')}</option>
- <option value="never">{t('settings.archiveMode.never')}</option>
- <option value="ask">{t('settings.archiveMode.ask')}</option>
- </select>
- <p className="text-xs text-bambu-gray mt-1">
- When printing from File Manager, optionally create an archive entry
- </p>
- </div>
- {/* Disk Space Warning Threshold */}
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Low Disk Space Warning
- </label>
- <div className="flex items-center gap-2">
- <input
- type="number"
- min="0.5"
- max="100"
- step="0.5"
- value={localSettings.library_disk_warning_gb ?? 5}
- onChange={(e) => updateSetting('library_disk_warning_gb', parseFloat(e.target.value) || 5)}
- 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"
- />
- <span className="text-bambu-gray">GB</span>
- </div>
- <p className="text-xs text-bambu-gray mt-1">
- Show warning when free disk space falls below this threshold
- </p>
- </div>
- </CardContent>
- </Card>
- </div>
- {/* Third Column - Sidebar Links & Updates */}
- <div className="space-y-6 flex-1 lg:max-w-sm">
- {/* Sidebar Links */}
- <ExternalLinksSettings />
- <Card>
- <CardHeader>
- <h2 className="text-lg font-semibold text-white">Updates</h2>
- </CardHeader>
- <CardContent className="space-y-4">
- <div className="flex items-center justify-between">
- <div>
- <p className="text-white">{t('settings.checkForUpdatesLabel')}</p>
- <p className="text-sm text-bambu-gray">
- Automatically check for new versions on startup
- </p>
- </div>
- <label className="relative inline-flex items-center cursor-pointer">
- <input
- type="checkbox"
- checked={localSettings.check_updates}
- onChange={(e) => updateSetting('check_updates', e.target.checked)}
- className="sr-only peer"
- />
- <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>
- </label>
- </div>
- <div className="flex items-center justify-between">
- <div>
- <p className="text-white">{t('settings.checkPrinterFirmware')}</p>
- <p className="text-sm text-bambu-gray">
- Check for printer firmware updates from Bambu Lab
- </p>
- </div>
- <label className="relative inline-flex items-center cursor-pointer">
- <input
- type="checkbox"
- checked={localSettings.check_printer_firmware ?? true}
- onChange={(e) => updateSetting('check_printer_firmware', e.target.checked)}
- className="sr-only peer"
- />
- <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>
- </label>
- </div>
- <div className="border-t border-bambu-dark-tertiary pt-4">
- <div className="flex items-center justify-between mb-2">
- <div>
- <p className="text-white">{t('settings.currentVersion')}</p>
- <p className="text-sm text-bambu-gray">v{versionInfo?.version || '...'}</p>
- </div>
- <Button
- variant="secondary"
- size="sm"
- onClick={() => refetchUpdateCheck()}
- disabled={isCheckingUpdate}
- >
- {isCheckingUpdate ? (
- <Loader2 className="w-4 h-4 animate-spin" />
- ) : (
- <RefreshCw className="w-4 h-4" />
- )}
- Check now
- </Button>
- </div>
- {updateCheck?.update_available ? (
- <div className="mt-4 p-3 bg-bambu-green/10 border border-bambu-green/30 rounded-lg">
- <div className="flex items-start justify-between">
- <div>
- <p className="text-bambu-green font-medium">
- Update available: v{updateCheck.latest_version}
- </p>
- {updateCheck.release_name && updateCheck.release_name !== updateCheck.latest_version && (
- <p className="text-sm text-bambu-gray mt-1">{updateCheck.release_name}</p>
- )}
- </div>
- <div className="flex items-center gap-2">
- {updateCheck.release_notes && (
- <button
- onClick={() => setShowReleaseNotes(true)}
- className="text-bambu-gray hover:text-white transition-colors text-sm underline"
- >
- Release Notes
- </button>
- )}
- {updateCheck.release_url && (
- <a
- href={updateCheck.release_url}
- target="_blank"
- rel="noopener noreferrer"
- className="text-bambu-gray hover:text-white transition-colors"
- title={t('settings.viewReleaseOnGitHub')}
- >
- <ExternalLink className="w-4 h-4" />
- </a>
- )}
- </div>
- </div>
- {updateStatus?.status === 'downloading' || updateStatus?.status === 'installing' ? (
- <div className="mt-3">
- <div className="flex items-center gap-2 text-sm text-bambu-gray">
- <Loader2 className="w-4 h-4 animate-spin" />
- <span>{updateStatus.message}</span>
- </div>
- <div className="mt-2 w-full bg-bambu-dark-tertiary rounded-full h-2">
- <div
- className="bg-bambu-green h-2 rounded-full transition-all duration-300"
- style={{ width: `${updateStatus.progress}%` }}
- />
- </div>
- </div>
- ) : updateStatus?.status === 'complete' ? (
- <div className="mt-3 p-2 bg-bambu-green/20 rounded text-sm text-bambu-green">
- {updateStatus.message}
- </div>
- ) : updateStatus?.status === 'error' ? (
- <div className="mt-3 p-2 bg-red-500/20 rounded text-sm text-red-400">
- {updateStatus.error || updateStatus.message}
- </div>
- ) : updateCheck?.is_docker ? (
- <div className="mt-3 p-3 bg-bambu-dark-tertiary rounded-lg">
- <p className="text-sm text-bambu-gray mb-2">
- Update via Docker Compose:
- </p>
- <code className="block text-xs bg-bambu-dark p-2 rounded text-bambu-green font-mono">
- docker compose pull && docker compose up -d
- </code>
- </div>
- ) : (
- <Button
- className="mt-3"
- onClick={() => applyUpdateMutation.mutate()}
- disabled={applyUpdateMutation.isPending}
- >
- {applyUpdateMutation.isPending ? (
- <Loader2 className="w-4 h-4 animate-spin" />
- ) : (
- <Download className="w-4 h-4" />
- )}
- Install Update
- </Button>
- )}
- </div>
- ) : updateCheck?.error ? (
- <div className="mt-2 p-2 bg-red-500/10 border border-red-500/30 rounded text-sm text-red-400">
- Failed to check for updates: {updateCheck.error}
- </div>
- ) : updateCheck && !updateCheck.update_available ? (
- <p className="mt-2 text-sm text-bambu-gray">
- You're running the latest version
- </p>
- ) : null}
- </div>
- </CardContent>
- </Card>
- {/* Data Management */}
- <Card>
- <CardHeader>
- <h2 className="text-lg font-semibold text-white">{t('settings.dataManagement')}</h2>
- </CardHeader>
- <CardContent className="space-y-4">
- <div className="flex items-center justify-between">
- <div>
- <p className="text-white">{t('settings.clearNotificationLogs')}</p>
- <p className="text-sm text-bambu-gray">
- {t('settings.clearNotificationLogsDescription')}
- </p>
- </div>
- <Button
- variant="secondary"
- size="sm"
- onClick={() => setShowClearLogsConfirm(true)}
- >
- <Trash2 className="w-4 h-4" />
- {t('common.clear')}
- </Button>
- </div>
- <div className="flex items-center justify-between">
- <div>
- <p className="text-white">{t('settings.resetUiPreferences')}</p>
- <p className="text-sm text-bambu-gray">
- {t('settings.resetUiPreferencesDescription')}
- </p>
- </div>
- <Button
- variant="secondary"
- size="sm"
- onClick={() => setShowClearStorageConfirm(true)}
- >
- <Trash2 className="w-4 h-4" />
- Reset
- </Button>
- </div>
- <div className="flex items-center justify-between pt-4 border-t border-bambu-dark-tertiary">
- <div>
- <p className="text-white">Backup & Restore</p>
- <p className="text-sm text-bambu-gray">
- Export/import settings and configure GitHub backup
- </p>
- </div>
- <Button
- variant="secondary"
- size="sm"
- onClick={() => handleTabChange('backup')}
- >
- <Database className="w-4 h-4" />
- Go to Backup
- </Button>
- </div>
- </CardContent>
- </Card>
- </div>
- </div>
- )}
- {/* Network Tab */}
- {activeTab === 'network' && localSettings && (
- <div className="flex flex-col lg:flex-row gap-6">
- {/* Left Column - External URL & FTP Retry */}
- <div className="flex-1 lg:max-w-xl space-y-4">
- {/* External URL */}
- <Card>
- <CardHeader>
- <h2 className="text-lg font-semibold text-white flex items-center gap-2">
- <Globe className="w-5 h-5 text-blue-400" />
- External URL
- </h2>
- </CardHeader>
- <CardContent className="space-y-4">
- <p className="text-sm text-bambu-gray">
- The external URL where Bambuddy is accessible. Used for notification images and external integrations.
- </p>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Bambuddy URL
- </label>
- <input
- type="text"
- value={localSettings.external_url ?? ''}
- onChange={(e) => updateSetting('external_url', e.target.value)}
- placeholder="http://192.168.1.100:8000"
- 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"
- />
- <p className="text-xs text-bambu-gray mt-1">
- Include protocol and port (e.g., http://192.168.1.100:8000)
- </p>
- </div>
- </CardContent>
- </Card>
- <Card>
- <CardHeader>
- <h2 className="text-lg font-semibold text-white flex items-center gap-2">
- <RefreshCw className="w-5 h-5 text-blue-400" />
- FTP Retry
- </h2>
- </CardHeader>
- <CardContent className="space-y-4">
- <p className="text-sm text-bambu-gray">
- Retry FTP operations when printer WiFi is unreliable. Applies to 3MF downloads, print uploads, timelapse downloads, and firmware updates.
- </p>
- <div className="flex items-center justify-between">
- <div>
- <p className="text-white">{t('settings.enableRetry')}</p>
- <p className="text-sm text-bambu-gray">
- Automatically retry failed FTP operations
- </p>
- </div>
- <label className="relative inline-flex items-center cursor-pointer">
- <input
- type="checkbox"
- checked={localSettings.ftp_retry_enabled ?? true}
- onChange={(e) => updateSetting('ftp_retry_enabled', e.target.checked)}
- className="sr-only peer"
- />
- <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>
- </label>
- </div>
- {localSettings.ftp_retry_enabled && (
- <div className="space-y-4 pt-2 border-t border-bambu-dark-tertiary">
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Retry attempts
- </label>
- <div className="relative w-44">
- <select
- value={localSettings.ftp_retry_count ?? 3}
- onChange={(e) => updateSetting('ftp_retry_count', parseInt(e.target.value))}
- 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"
- >
- {[1, 2, 3, 4, 5, 6, 7, 8, 9, 10].map(n => (
- <option key={n} value={n}>{n} {n === 1 ? 'time' : 'times'}</option>
- ))}
- </select>
- <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
- </div>
- </div>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Retry delay
- </label>
- <div className="relative w-44">
- <select
- value={localSettings.ftp_retry_delay ?? 2}
- onChange={(e) => updateSetting('ftp_retry_delay', parseInt(e.target.value))}
- 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"
- >
- {[1, 2, 3, 5, 10, 15, 20, 30].map(n => (
- <option key={n} value={n}>{n} {n === 1 ? 'second' : 'seconds'}</option>
- ))}
- </select>
- <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
- </div>
- </div>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Connection timeout
- </label>
- <div className="relative w-44">
- <select
- value={localSettings.ftp_timeout ?? 30}
- onChange={(e) => updateSetting('ftp_timeout', parseInt(e.target.value))}
- 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"
- >
- {[10, 15, 20, 30, 45, 60, 90, 120].map(n => (
- <option key={n} value={n}>{n} seconds</option>
- ))}
- </select>
- <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
- </div>
- <p className="text-xs text-bambu-gray mt-1">
- Increase for printers with weak WiFi
- </p>
- </div>
- </div>
- )}
- </CardContent>
- </Card>
- </div>
- {/* Right Column - Home Assistant & MQTT Publishing */}
- <div className="flex-1 lg:max-w-xl space-y-4">
- {/* Home Assistant Integration */}
- <Card>
- <CardHeader>
- <div className="flex items-center justify-between">
- <h2 className="text-lg font-semibold text-white flex items-center gap-2">
- <Home className="w-5 h-5 text-bambu-green" />
- Home Assistant
- </h2>
- {localSettings.ha_enabled && haTestResult && (
- <div className="flex items-center gap-2">
- <span className={`w-2.5 h-2.5 rounded-full ${haTestResult.success ? 'bg-green-400' : 'bg-red-400'}`} />
- <span className={`text-sm ${haTestResult.success ? 'text-green-400' : 'text-red-400'}`}>
- {haTestResult.success ? 'Connected' : 'Disconnected'}
- </span>
- </div>
- )}
- </div>
- </CardHeader>
- <CardContent className="space-y-4">
- <p className="text-sm text-bambu-gray">
- Connect to Home Assistant to control smart plugs via HA's REST API. Supports switch, light, input_boolean, and script entities.
- </p>
- <div className="flex items-center justify-between">
- <div>
- <p className="text-white">{t('settings.enableHomeAssistant')}</p>
- <p className="text-xs text-bambu-gray">{t('settings.homeAssistantDescription')}</p>
- </div>
- <label className="relative inline-flex items-center cursor-pointer">
- <input
- type="checkbox"
- checked={localSettings.ha_enabled ?? false}
- onChange={(e) => updateSetting('ha_enabled', e.target.checked)}
- className="sr-only peer"
- />
- <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>
- </label>
- </div>
- {localSettings.ha_enabled && (
- <>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Home Assistant URL
- </label>
- <input
- type="text"
- value={localSettings.ha_url ?? ''}
- onChange={(e) => updateSetting('ha_url', e.target.value)}
- placeholder="http://192.168.1.100:8123"
- 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"
- />
- </div>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Long-Lived Access Token
- </label>
- <input
- type="password"
- value={localSettings.ha_token ?? ''}
- onChange={(e) => updateSetting('ha_token', e.target.value)}
- placeholder="eyJ0eXAiOiJKV1QiLC..."
- 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"
- />
- <p className="text-xs text-bambu-gray mt-1">
- Create a token in HA: Profile → Long-Lived Access Tokens → Create Token
- </p>
- </div>
- {localSettings.ha_url && localSettings.ha_token && (
- <div className="pt-2 border-t border-bambu-dark-tertiary">
- <Button
- variant="secondary"
- size="sm"
- disabled={haTestLoading}
- onClick={async () => {
- setHaTestLoading(true);
- setHaTestResult(null);
- try {
- const result = await api.testHAConnection(localSettings.ha_url!, localSettings.ha_token!);
- setHaTestResult(result);
- } catch (e) {
- setHaTestResult({ success: false, message: null, error: e instanceof Error ? e.message : t('common.unknownError') });
- } finally {
- setHaTestLoading(false);
- }
- }}
- >
- {haTestLoading ? <Loader2 className="w-4 h-4 animate-spin" /> : <Wifi className="w-4 h-4" />}
- {t('settings.testConnection')}
- </Button>
- </div>
- )}
- </>
- )}
- </CardContent>
- </Card>
- {/* MQTT Publishing */}
- <Card>
- <CardHeader>
- <div className="flex items-center justify-between">
- <h2 className="text-lg font-semibold text-white flex items-center gap-2">
- <Wifi className="w-5 h-5 text-blue-400" />
- MQTT Publishing
- </h2>
- {mqttStatus?.enabled && (
- <div className="flex items-center gap-2">
- <span className={`w-2.5 h-2.5 rounded-full ${mqttStatus.connected ? 'bg-green-400' : 'bg-red-400'}`} />
- <span className={`text-sm ${mqttStatus.connected ? 'text-green-400' : 'text-red-400'}`}>
- {mqttStatus.connected ? 'Connected' : 'Disconnected'}
- </span>
- </div>
- )}
- </div>
- </CardHeader>
- <CardContent className="space-y-4">
- <p className="text-sm text-bambu-gray">
- Publish BamBuddy events to an external MQTT broker for integration with Node-RED, Home Assistant, and other automation systems.
- </p>
- <div className="flex items-center justify-between">
- <div>
- <p className="text-white">{t('settings.enableMqtt')}</p>
- <p className="text-sm text-bambu-gray">
- Publish events to external MQTT broker
- </p>
- </div>
- <label className="relative inline-flex items-center cursor-pointer">
- <input
- type="checkbox"
- checked={localSettings.mqtt_enabled ?? false}
- onChange={(e) => updateSetting('mqtt_enabled', e.target.checked)}
- className="sr-only peer"
- />
- <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>
- </label>
- </div>
- {localSettings.mqtt_enabled && (
- <div className="space-y-4 pt-2 border-t border-bambu-dark-tertiary">
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Broker hostname
- </label>
- <input
- type="text"
- value={localSettings.mqtt_broker ?? ''}
- onChange={(e) => updateSetting('mqtt_broker', e.target.value)}
- placeholder="mqtt.example.com or 192.168.1.100"
- 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"
- />
- </div>
- <div className="flex items-end gap-4">
- <div className="flex-1">
- <label className="block text-sm text-bambu-gray mb-1">
- Port
- </label>
- <input
- type="number"
- min="1"
- max="65535"
- value={localSettings.mqtt_port ?? 1883}
- onChange={(e) => updateSetting('mqtt_port', Math.min(65535, Math.max(1, parseInt(e.target.value) || 1883)))}
- 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"
- />
- </div>
- <div className="flex items-center gap-3 pb-2">
- <label className="relative inline-flex items-center cursor-pointer">
- <input
- type="checkbox"
- checked={localSettings.mqtt_use_tls ?? false}
- onChange={(e) => {
- const useTls = e.target.checked;
- updateSetting('mqtt_use_tls', useTls);
- // Auto-populate port based on TLS selection
- const currentPort = localSettings.mqtt_port ?? 1883;
- if (useTls && currentPort === 1883) {
- updateSetting('mqtt_port', 8883);
- } else if (!useTls && currentPort === 8883) {
- updateSetting('mqtt_port', 1883);
- }
- }}
- className="sr-only peer"
- />
- <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>
- </label>
- <span className="text-white text-sm">{t('settings.useTls')}</span>
- </div>
- </div>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Username (optional)
- </label>
- <input
- type="text"
- value={localSettings.mqtt_username ?? ''}
- onChange={(e) => updateSetting('mqtt_username', e.target.value)}
- placeholder={t('settings.leaveEmptyForAnonymous')}
- 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"
- />
- </div>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Password (optional)
- </label>
- <input
- type="password"
- value={localSettings.mqtt_password ?? ''}
- onChange={(e) => updateSetting('mqtt_password', e.target.value)}
- placeholder={t('settings.leaveEmptyForAnonymous')}
- 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"
- />
- </div>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Topic prefix
- </label>
- <input
- type="text"
- value={localSettings.mqtt_topic_prefix ?? 'bambuddy'}
- onChange={(e) => updateSetting('mqtt_topic_prefix', e.target.value)}
- placeholder="bambuddy"
- 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"
- />
- <p className="text-xs text-bambu-gray mt-1">
- Topics will be: {localSettings.mqtt_topic_prefix || 'bambuddy'}/printers/<serial>/status, etc.
- </p>
- </div>
- {/* Connection Info */}
- {mqttStatus && (
- <div className="pt-3 mt-3 border-t border-bambu-dark-tertiary">
- <div className="flex items-center gap-2 text-sm">
- <span className={`w-2 h-2 rounded-full ${mqttStatus.connected ? 'bg-green-400' : 'bg-red-400'}`} />
- <span className="text-bambu-gray">
- {mqttStatus.connected ? (
- <>{t('settings.mqttConnectedTo')} <span className="text-white">{mqttStatus.broker}:{mqttStatus.port}</span></>
- ) : (
- t('settings.spoolmanDisconnected')
- )}
- </span>
- </div>
- </div>
- )}
- </div>
- )}
- </CardContent>
- </Card>
- </div>
- {/* Third Column - Prometheus Metrics */}
- <div className="flex-1 lg:max-w-md space-y-4">
- <Card>
- <CardHeader>
- <h2 className="text-lg font-semibold text-white flex items-center gap-2">
- <TrendingUp className="w-5 h-5 text-orange-400" />
- Prometheus Metrics
- </h2>
- </CardHeader>
- <CardContent className="space-y-4">
- <p className="text-sm text-bambu-gray">
- Expose printer metrics at <code className="bg-bambu-dark px-1 rounded">/api/v1/metrics</code> for Prometheus/Grafana monitoring.
- </p>
- <div className="flex items-center justify-between">
- <div>
- <p className="text-white">{t('settings.enableMetricsEndpoint')}</p>
- <p className="text-xs text-bambu-gray">{t('settings.prometheusDescription')}</p>
- </div>
- <label className="relative inline-flex items-center cursor-pointer">
- <input
- type="checkbox"
- checked={localSettings.prometheus_enabled ?? false}
- onChange={(e) => updateSetting('prometheus_enabled', e.target.checked)}
- className="sr-only peer"
- />
- <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>
- </label>
- </div>
- {localSettings.prometheus_enabled && (
- <div className="space-y-4 pt-2 border-t border-bambu-dark-tertiary">
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- Bearer Token (optional)
- </label>
- <input
- type="password"
- value={localSettings.prometheus_token ?? ''}
- onChange={(e) => updateSetting('prometheus_token', e.target.value)}
- placeholder={t('settings.leaveEmptyForNoAuth')}
- 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"
- />
- <p className="text-xs text-bambu-gray mt-1">
- If set, requests must include <code className="bg-bambu-dark px-1 rounded">Authorization: Bearer <token></code>
- </p>
- </div>
- <div className="pt-2 border-t border-bambu-dark-tertiary">
- <p className="text-sm text-white mb-2">{t('settings.availableMetrics')}</p>
- <div className="text-xs text-bambu-gray space-y-1">
- <p><code className="text-orange-400">bambuddy_printer_connected</code> - Connection status</p>
- <p><code className="text-orange-400">bambuddy_printer_state</code> - Printer state (idle/printing/etc)</p>
- <p><code className="text-orange-400">bambuddy_print_progress</code> - Print progress 0-100%</p>
- <p><code className="text-orange-400">bambuddy_bed_temp_celsius</code> - Bed temperature</p>
- <p><code className="text-orange-400">bambuddy_nozzle_temp_celsius</code> - Nozzle temperature</p>
- <p><code className="text-orange-400">bambuddy_prints_total</code> - Total prints by result</p>
- <p className="text-bambu-gray/70 italic">...and more (layers, fans, queue, filament usage)</p>
- </div>
- </div>
- </div>
- )}
- </CardContent>
- </Card>
- </div>
- </div>
- )}
- {/* Home Assistant Test Connection Modal */}
- {haTestResult && (
- <div className="fixed inset-0 bg-black/50 flex items-center justify-center z-50">
- <div className="bg-bambu-dark-secondary rounded-lg p-6 max-w-md w-full mx-4">
- <div className="flex items-center gap-3 mb-4">
- {haTestResult.success ? (
- <CheckCircle className="w-8 h-8 text-green-400" />
- ) : (
- <XCircle className="w-8 h-8 text-red-400" />
- )}
- <h3 className="text-lg font-medium text-white">
- {haTestResult.success ? 'Connection Successful' : 'Connection Failed'}
- </h3>
- </div>
- <p className="text-bambu-gray mb-6">
- {haTestResult.success
- ? haTestResult.message || 'Successfully connected to Home Assistant.'
- : haTestResult.error || 'Failed to connect to Home Assistant.'}
- </p>
- <div className="flex justify-end">
- <Button
- variant="primary"
- onClick={() => setHaTestResult(null)}
- >
- OK
- </Button>
- </div>
- </div>
- </div>
- )}
- {/* Smart Plugs Tab */}
- {activeTab === 'plugs' && (
- <div className="max-w-4xl">
- <div className="flex items-start justify-between mb-6">
- <div>
- <h2 className="text-lg font-semibold text-white flex items-center gap-2">
- <Plug className="w-5 h-5 text-bambu-green" />
- Smart Plugs
- </h2>
- <p className="text-sm text-bambu-gray mt-1">
- Connect smart plugs (Tasmota or Home Assistant) to automate power control and track energy usage for your printers.
- </p>
- </div>
- <div className="flex items-center gap-2 pt-1 shrink-0">
- {smartPlugs && smartPlugs.filter(p => p.enabled).length > 1 && (
- <>
- <Button
- variant="secondary"
- size="sm"
- className="whitespace-nowrap"
- onClick={() => setShowBulkPlugConfirm('on')}
- disabled={bulkPlugActionMutation.isPending}
- title={t('settings.turnAllPlugsOn')}
- >
- {bulkPlugActionMutation.isPending ? (
- <Loader2 className="w-4 h-4 animate-spin" />
- ) : (
- <Power className="w-4 h-4 text-bambu-green" />
- )}
- All On
- </Button>
- <Button
- variant="secondary"
- size="sm"
- className="whitespace-nowrap"
- onClick={() => setShowBulkPlugConfirm('off')}
- disabled={bulkPlugActionMutation.isPending}
- title={t('settings.turnAllPlugsOff')}
- >
- {bulkPlugActionMutation.isPending ? (
- <Loader2 className="w-4 h-4 animate-spin" />
- ) : (
- <PowerOff className="w-4 h-4 text-red-400" />
- )}
- All Off
- </Button>
- </>
- )}
- <Button
- className="whitespace-nowrap"
- onClick={() => {
- setEditingPlug(null);
- setShowPlugModal(true);
- }}
- >
- <Plus className="w-4 h-4" />
- Add Smart Plug
- </Button>
- </div>
- </div>
- {/* Energy Summary Card */}
- {smartPlugs && smartPlugs.length > 0 && (
- <Card className="mb-6">
- <CardHeader>
- <h3 className="text-base font-semibold text-white flex items-center gap-2">
- <Zap className="w-4 h-4 text-yellow-400" />
- Energy Summary
- {energyLoading && (
- <Loader2 className="w-4 h-4 animate-spin text-bambu-gray ml-2" />
- )}
- </h3>
- </CardHeader>
- <CardContent>
- {plugEnergySummary ? (
- <div className="grid grid-cols-2 md:grid-cols-4 gap-4">
- {/* Current Power */}
- <div className="bg-bambu-dark rounded-lg p-3">
- <div className="flex items-center gap-2 text-bambu-gray text-xs mb-1">
- <Zap className="w-3 h-3" />
- Current Power
- </div>
- <div className="text-xl font-bold text-white">
- {plugEnergySummary.totalPower.toFixed(1)}
- <span className="text-sm font-normal text-bambu-gray ml-1">W</span>
- </div>
- <div className="text-xs text-bambu-gray mt-1">
- {plugEnergySummary.reachableCount}/{plugEnergySummary.totalPlugs} plugs online
- </div>
- </div>
- {/* Today */}
- <div className="bg-bambu-dark rounded-lg p-3">
- <div className="flex items-center gap-2 text-bambu-gray text-xs mb-1">
- <Calendar className="w-3 h-3" />
- Today
- </div>
- <div className="text-xl font-bold text-white">
- {plugEnergySummary.totalToday.toFixed(2)}
- <span className="text-sm font-normal text-bambu-gray ml-1">kWh</span>
- </div>
- {(localSettings?.energy_cost_per_kwh ?? 0) > 0 && (
- <div className="text-xs text-bambu-gray mt-1">
- ~{(plugEnergySummary.totalToday * (localSettings?.energy_cost_per_kwh ?? 0)).toFixed(2)} {localSettings?.currency}
- </div>
- )}
- </div>
- {/* Yesterday */}
- <div className="bg-bambu-dark rounded-lg p-3">
- <div className="flex items-center gap-2 text-bambu-gray text-xs mb-1">
- <TrendingUp className="w-3 h-3" />
- Yesterday
- </div>
- <div className="text-xl font-bold text-white">
- {plugEnergySummary.totalYesterday.toFixed(2)}
- <span className="text-sm font-normal text-bambu-gray ml-1">kWh</span>
- </div>
- {(localSettings?.energy_cost_per_kwh ?? 0) > 0 && (
- <div className="text-xs text-bambu-gray mt-1">
- ~{(plugEnergySummary.totalYesterday * (localSettings?.energy_cost_per_kwh ?? 0)).toFixed(2)} {localSettings?.currency}
- </div>
- )}
- </div>
- {/* Total Lifetime */}
- <div className="bg-bambu-dark rounded-lg p-3">
- <div className="flex items-center gap-2 text-bambu-gray text-xs mb-1">
- <DollarSign className="w-3 h-3" />
- Total
- </div>
- <div className="text-xl font-bold text-white">
- {plugEnergySummary.totalLifetime.toFixed(1)}
- <span className="text-sm font-normal text-bambu-gray ml-1">kWh</span>
- </div>
- {(localSettings?.energy_cost_per_kwh ?? 0) > 0 && (
- <div className="text-xs text-bambu-gray mt-1">
- ~{(plugEnergySummary.totalLifetime * (localSettings?.energy_cost_per_kwh ?? 0)).toFixed(2)} {localSettings?.currency}
- </div>
- )}
- </div>
- </div>
- ) : !energyLoading ? (
- <p className="text-sm text-bambu-gray">
- Enable plugs to see energy summary
- </p>
- ) : null}
- </CardContent>
- </Card>
- )}
- {plugsLoading ? (
- <div className="flex justify-center py-12">
- <Loader2 className="w-8 h-8 text-bambu-green animate-spin" />
- </div>
- ) : smartPlugs && smartPlugs.length > 0 ? (
- <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
- {smartPlugs.map((plug) => (
- <SmartPlugCard
- key={plug.id}
- plug={plug}
- onEdit={(p) => {
- setEditingPlug(p);
- setShowPlugModal(true);
- }}
- />
- ))}
- </div>
- ) : (
- <Card>
- <CardContent className="py-12">
- <div className="text-center text-bambu-gray">
- <Plug className="w-16 h-16 mx-auto mb-4 opacity-30" />
- <p className="text-lg font-medium text-white mb-2">{t('settings.noSmartPlugsTitle')}</p>
- <p className="text-sm mb-4">{t('settings.noSmartPlugsDescription')}</p>
- <Button
- onClick={() => {
- setEditingPlug(null);
- setShowPlugModal(true);
- }}
- >
- <Plus className="w-4 h-4" />
- {t('settings.addFirstSmartPlug')}
- </Button>
- </div>
- </CardContent>
- </Card>
- )}
- </div>
- )}
- {/* Notifications Tab */}
- {activeTab === 'notifications' && (
- <div className="grid grid-cols-1 lg:grid-cols-2 gap-8">
- {/* Left Column: Providers */}
- <div>
- <div className="flex items-center justify-between mb-4">
- <h2 className="text-lg font-semibold text-white flex items-center gap-2">
- <Bell className="w-5 h-5 text-bambu-green" />
- {t('settings.providers')}
- </h2>
- <div className="flex items-center gap-2">
- <Button
- size="sm"
- variant="secondary"
- onClick={() => setShowLogViewer(true)}
- >
- <History className="w-4 h-4" />
- {t('settings.log')}
- </Button>
- {notificationProviders && notificationProviders.length > 0 && (
- <Button
- size="sm"
- variant="secondary"
- onClick={() => {
- setTestAllResult(null);
- testAllMutation.mutate();
- }}
- disabled={testAllMutation.isPending}
- >
- {testAllMutation.isPending ? (
- <Loader2 className="w-4 h-4 animate-spin" />
- ) : (
- <Send className="w-4 h-4" />
- )}
- {t('settings.testAll')}
- </Button>
- )}
- <Button
- size="sm"
- onClick={() => {
- setEditingProvider(null);
- setShowNotificationModal(true);
- }}
- >
- <Plus className="w-4 h-4" />
- Add
- </Button>
- </div>
- </div>
- {/* Notification Language Setting */}
- <Card className="mb-4">
- <CardContent className="py-3">
- <div className="flex items-center justify-between">
- <div>
- <p className="text-white text-sm font-medium">{t('settings.notificationLanguage')}</p>
- <p className="text-xs text-bambu-gray">{t('settings.notificationLanguageDescription')}</p>
- </div>
- <select
- value={localSettings.notification_language || 'en'}
- onChange={(e) => updateSetting('notification_language', e.target.value)}
- 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"
- >
- {availableLanguages.map((lang) => (
- <option key={lang.code} value={lang.code}>
- {lang.nativeName}
- </option>
- ))}
- </select>
- </div>
- </CardContent>
- </Card>
- {/* Test All Results */}
- {testAllResult && (
- <Card className="mb-4">
- <CardContent className="py-3">
- <div className="flex items-center justify-between mb-2">
- <span className="text-sm font-medium text-white">{t('settings.testResults')}</span>
- <button
- onClick={() => setTestAllResult(null)}
- className="text-bambu-gray hover:text-white text-xs"
- >
- {t('common.dismiss')}
- </button>
- </div>
- <div className="flex items-center gap-4 text-sm mb-2">
- <span className="flex items-center gap-1 text-bambu-green">
- <CheckCircle className="w-4 h-4" />
- {t('settings.testPassedCount', { count: testAllResult.success })}
- </span>
- {testAllResult.failed > 0 && (
- <span className="flex items-center gap-1 text-red-400">
- <XCircle className="w-4 h-4" />
- {t('settings.testFailedCount', { count: testAllResult.failed })}
- </span>
- )}
- </div>
- {testAllResult.results.filter(r => !r.success).length > 0 && (
- <div className="space-y-1 mt-2 pt-2 border-t border-bambu-dark-tertiary">
- {testAllResult.results.filter(r => !r.success).map((result) => (
- <div key={result.provider_id} className="text-xs text-red-400">
- <span className="font-medium">{result.provider_name}:</span> {result.message}
- </div>
- ))}
- </div>
- )}
- </CardContent>
- </Card>
- )}
- {providersLoading ? (
- <div className="flex justify-center py-12">
- <Loader2 className="w-6 h-6 text-bambu-green animate-spin" />
- </div>
- ) : notificationProviders && notificationProviders.length > 0 ? (
- <div className="space-y-3">
- {notificationProviders.map((provider) => (
- <NotificationProviderCard
- key={provider.id}
- provider={provider}
- onEdit={(p) => {
- setEditingProvider(p);
- setShowNotificationModal(true);
- }}
- />
- ))}
- </div>
- ) : (
- <Card>
- <CardContent className="py-8">
- <div className="text-center text-bambu-gray">
- <Bell className="w-12 h-12 mx-auto mb-3 opacity-30" />
- <p className="text-sm font-medium text-white mb-2">{t('settings.noProvidersTitle')}</p>
- <p className="text-xs mb-3">{t('settings.noProvidersDescription')}</p>
- <Button
- size="sm"
- onClick={() => {
- setEditingProvider(null);
- setShowNotificationModal(true);
- }}
- >
- <Plus className="w-4 h-4" />
- {t('settings.addProvider')}
- </Button>
- </div>
- </CardContent>
- </Card>
- )}
- </div>
- {/* Right Column: Templates */}
- <div>
- <h2 className="text-lg font-semibold text-white flex items-center gap-2 mb-4">
- <FileText className="w-5 h-5 text-bambu-green" />
- {t('settings.messageTemplates')}
- </h2>
- <p className="text-sm text-bambu-gray mb-4">
- {t('settings.messageTemplatesDescription')}
- </p>
- {templatesLoading ? (
- <div className="flex justify-center py-8">
- <Loader2 className="w-6 h-6 text-bambu-green animate-spin" />
- </div>
- ) : notificationTemplates && notificationTemplates.length > 0 ? (
- <div className="space-y-2">
- {notificationTemplates.map((template) => (
- <Card
- key={template.id}
- className="cursor-pointer hover:border-bambu-green/50 transition-colors"
- onClick={() => setEditingTemplate(template)}
- >
- <CardContent className="py-2.5 px-3">
- <div className="flex items-center justify-between">
- <div className="min-w-0 flex-1">
- <p className="text-white font-medium text-sm truncate">{template.name}</p>
- <p className="text-bambu-gray text-xs truncate mt-0.5">
- {template.title_template}
- </p>
- </div>
- <button
- className="p-1.5 hover:bg-bambu-dark-tertiary rounded transition-colors shrink-0 ml-2"
- onClick={(e) => {
- e.stopPropagation();
- setEditingTemplate(template);
- }}
- >
- <Edit2 className="w-4 h-4 text-bambu-gray" />
- </button>
- </div>
- </CardContent>
- </Card>
- ))}
- </div>
- ) : (
- <Card>
- <CardContent className="py-8">
- <div className="text-center text-bambu-gray">
- <FileText className="w-12 h-12 mx-auto mb-3 opacity-30" />
- <p className="text-sm">{t('settings.noTemplatesAvailable')}</p>
- </div>
- </CardContent>
- </Card>
- )}
- </div>
- </div>
- )}
- {/* API Keys Tab */}
- {activeTab === 'apikeys' && (
- <div className="grid grid-cols-1 xl:grid-cols-2 gap-8">
- {/* Left Column - API Keys Management */}
- <div>
- <div className="flex items-start justify-between gap-4 mb-6">
- <div className="flex-1">
- <h2 className="text-lg font-semibold text-white flex items-center gap-2">
- <Key className="w-5 h-5 text-bambu-green" />
- {t('settings.apiKeys')}
- </h2>
- <p className="text-sm text-bambu-gray mt-1">
- {t('settings.apiKeysDescription')}
- </p>
- </div>
- <Button size="sm" onClick={() => setShowCreateAPIKey(true)} className="flex-shrink-0">
- <Plus className="w-4 h-4" />
- {t('settings.createKey')}
- </Button>
- </div>
- {/* Created Key Display */}
- {createdAPIKey && (
- <Card className="mb-6 border-bambu-green">
- <CardContent className="py-4">
- <div className="flex items-start gap-3">
- <CheckCircle className="w-5 h-5 text-bambu-green flex-shrink-0 mt-0.5" />
- <div className="flex-1">
- <p className="text-white font-medium mb-1">{t('settings.apiKeyCreated')}</p>
- <p className="text-sm text-bambu-gray mb-2">
- {t('settings.apiKeyCopyWarning')}
- </p>
- <div className="flex items-center gap-2 bg-bambu-dark rounded-lg p-2">
- <code className="flex-1 text-sm text-bambu-green font-mono break-all">
- {createdAPIKey}
- </code>
- <Button
- variant="secondary"
- size="sm"
- onClick={async () => {
- try {
- if (navigator.clipboard && navigator.clipboard.writeText) {
- await navigator.clipboard.writeText(createdAPIKey);
- } else {
- const textArea = document.createElement('textarea');
- textArea.value = createdAPIKey;
- textArea.style.position = 'fixed';
- textArea.style.left = '-999999px';
- document.body.appendChild(textArea);
- textArea.select();
- document.execCommand('copy');
- document.body.removeChild(textArea);
- }
- showToast(t('settings.toast.keyCopied'));
- } catch {
- showToast(t('settings.toast.copyFailed'), 'error');
- }
- }}
- >
- <Copy className="w-4 h-4" />
- </Button>
- </div>
- <div className="flex gap-2 mt-3">
- <Button
- variant="secondary"
- size="sm"
- onClick={() => {
- setTestApiKey(createdAPIKey);
- showToast(t('settings.toast.keyAddedToBrowser'));
- }}
- >
- {t('settings.useInApiBrowser')}
- </Button>
- <Button
- variant="secondary"
- size="sm"
- onClick={() => setCreatedAPIKey(null)}
- >
- {t('common.dismiss')}
- </Button>
- </div>
- </div>
- </div>
- </CardContent>
- </Card>
- )}
- {/* Create Key Form */}
- {showCreateAPIKey && (
- <Card className="mb-6">
- <CardHeader>
- <h3 className="text-base font-semibold text-white">{t('settings.createNewApiKey')}</h3>
- </CardHeader>
- <CardContent className="space-y-4">
- <div>
- <label className="block text-sm text-bambu-gray mb-1">{t('settings.keyName')}</label>
- <input
- type="text"
- value={newAPIKeyName}
- onChange={(e) => setNewAPIKeyName(e.target.value)}
- placeholder={t('settings.keyNamePlaceholder')}
- 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"
- />
- </div>
- <div>
- <label className="block text-sm text-bambu-gray mb-2">{t('common.permissions')}</label>
- <div className="space-y-2">
- <label className="flex items-center gap-3 cursor-pointer">
- <input
- type="checkbox"
- checked={newAPIKeyPermissions.can_read_status}
- onChange={(e) => setNewAPIKeyPermissions(prev => ({ ...prev, can_read_status: e.target.checked }))}
- className="w-4 h-4 text-bambu-green rounded border-bambu-dark-tertiary bg-bambu-dark focus:ring-bambu-green"
- />
- <div>
- <span className="text-white">{t('settings.readStatus')}</span>
- <p className="text-xs text-bambu-gray">{t('settings.readStatusDescription')}</p>
- </div>
- </label>
- <label className="flex items-center gap-3 cursor-pointer">
- <input
- type="checkbox"
- checked={newAPIKeyPermissions.can_queue}
- onChange={(e) => setNewAPIKeyPermissions(prev => ({ ...prev, can_queue: e.target.checked }))}
- className="w-4 h-4 text-bambu-green rounded border-bambu-dark-tertiary bg-bambu-dark focus:ring-bambu-green"
- />
- <div>
- <span className="text-white">{t('settings.manageQueue')}</span>
- <p className="text-xs text-bambu-gray">{t('settings.manageQueueDescription')}</p>
- </div>
- </label>
- <label className="flex items-center gap-3 cursor-pointer">
- <input
- type="checkbox"
- checked={newAPIKeyPermissions.can_control_printer}
- onChange={(e) => setNewAPIKeyPermissions(prev => ({ ...prev, can_control_printer: e.target.checked }))}
- className="w-4 h-4 text-bambu-green rounded border-bambu-dark-tertiary bg-bambu-dark focus:ring-bambu-green"
- />
- <div>
- <span className="text-white">{t('settings.controlPrinter')}</span>
- <p className="text-xs text-bambu-gray">{t('settings.controlPrinterDescription')}</p>
- </div>
- </label>
- </div>
- </div>
- <div className="flex items-center gap-2 pt-2">
- <Button
- onClick={() => createAPIKeyMutation.mutate({
- name: newAPIKeyName || t('settings.unnamedKey'),
- ...newAPIKeyPermissions,
- })}
- disabled={createAPIKeyMutation.isPending}
- >
- {createAPIKeyMutation.isPending ? (
- <Loader2 className="w-4 h-4 animate-spin" />
- ) : (
- <Plus className="w-4 h-4" />
- )}
- {t('settings.createKey')}
- </Button>
- <Button variant="secondary" onClick={() => setShowCreateAPIKey(false)}>
- {t('common.cancel')}
- </Button>
- </div>
- </CardContent>
- </Card>
- )}
- {/* Existing Keys List */}
- {apiKeysLoading ? (
- <div className="flex justify-center py-12">
- <Loader2 className="w-8 h-8 text-bambu-green animate-spin" />
- </div>
- ) : apiKeys && apiKeys.length > 0 ? (
- <div className="space-y-3">
- {apiKeys.map((key) => (
- <Card key={key.id}>
- <CardContent className="py-3">
- <div className="flex items-center justify-between">
- <div className="flex items-center gap-3">
- <Key className={`w-5 h-5 ${key.enabled ? 'text-bambu-green' : 'text-bambu-gray'}`} />
- <div>
- <p className="text-white font-medium">{key.name}</p>
- <p className="text-xs text-bambu-gray">
- {key.key_prefix}••••••••
- {key.last_used && ` · ${t('settings.lastUsed')}: ${formatDateOnly(key.last_used)}`}
- </p>
- </div>
- </div>
- <div className="flex items-center gap-2">
- <div className="flex gap-1 text-xs">
- {key.can_read_status && (
- <span className="px-1.5 py-0.5 bg-blue-500/20 text-blue-400 rounded">{t('settings.read')}</span>
- )}
- {key.can_queue && (
- <span className="px-1.5 py-0.5 bg-green-500/20 text-green-400 rounded">{t('queue.title')}</span>
- )}
- {key.can_control_printer && (
- <span className="px-1.5 py-0.5 bg-orange-500/20 text-orange-400 rounded">{t('settings.control')}</span>
- )}
- </div>
- <Button
- variant="secondary"
- size="sm"
- onClick={() => setShowDeleteAPIKeyConfirm(key.id)}
- >
- <Trash2 className="w-4 h-4 text-red-400" />
- </Button>
- </div>
- </div>
- </CardContent>
- </Card>
- ))}
- </div>
- ) : (
- <Card>
- <CardContent className="py-12">
- <div className="text-center text-bambu-gray">
- <Key className="w-16 h-16 mx-auto mb-4 opacity-30" />
- <p className="text-lg font-medium text-white mb-2">{t('settings.apiKeysEmptyTitle')}</p>
- <p className="text-sm mb-4">{t('settings.apiKeysEmptyDescription')}</p>
- <Button onClick={() => setShowCreateAPIKey(true)}>
- <Plus className="w-4 h-4" />
- {t('settings.createFirstKey')}
- </Button>
- </div>
- </CardContent>
- </Card>
- )}
- {/* Webhook Documentation */}
- <Card className="mt-6">
- <CardHeader>
- <h3 className="text-base font-semibold text-white">{t('settings.webhookEndpoints')}</h3>
- </CardHeader>
- <CardContent className="space-y-3 text-sm">
- <p className="text-bambu-gray">
- {t('settings.webhookApiKeyHint')}
- </p>
- <div className="space-y-2 font-mono text-xs">
- <div className="p-2 bg-bambu-dark rounded">
- <span className="text-blue-400">GET</span>{' '}
- <span className="text-white">/api/v1/webhook/status</span>
- <span className="text-bambu-gray"> - {t('settings.webhook.getAllStatus')}</span>
- </div>
- <div className="p-2 bg-bambu-dark rounded">
- <span className="text-blue-400">GET</span>{' '}
- <span className="text-white">/api/v1/webhook/status/:id</span>
- <span className="text-bambu-gray"> - {t('settings.webhook.getSpecificStatus')}</span>
- </div>
- <div className="p-2 bg-bambu-dark rounded">
- <span className="text-green-400">POST</span>{' '}
- <span className="text-white">/api/v1/webhook/queue</span>
- <span className="text-bambu-gray"> - {t('settings.webhook.addToQueue')}</span>
- </div>
- <div className="p-2 bg-bambu-dark rounded">
- <span className="text-orange-400">POST</span>{' '}
- <span className="text-white">/api/v1/webhook/printer/:id/pause</span>
- <span className="text-bambu-gray"> - {t('settings.webhook.pausePrint')}</span>
- </div>
- <div className="p-2 bg-bambu-dark rounded">
- <span className="text-orange-400">POST</span>{' '}
- <span className="text-white">/api/v1/webhook/printer/:id/resume</span>
- <span className="text-bambu-gray"> - {t('settings.webhook.resumePrint')}</span>
- </div>
- <div className="p-2 bg-bambu-dark rounded">
- <span className="text-red-400">POST</span>{' '}
- <span className="text-white">/api/v1/webhook/printer/:id/stop</span>
- <span className="text-bambu-gray"> - {t('settings.webhook.stopPrint')}</span>
- </div>
- </div>
- </CardContent>
- </Card>
- </div>
- {/* Right Column - API Browser */}
- <div>
- <div className="mb-6">
- <h2 className="text-lg font-semibold text-white flex items-center gap-2">
- <Globe className="w-5 h-5 text-bambu-green" />
- {t('settings.apiBrowser')}
- </h2>
- <p className="text-sm text-bambu-gray mt-1">
- {t('settings.apiBrowserDescription')}
- </p>
- </div>
- {/* API Key Input for Testing */}
- <Card className="mb-4">
- <CardContent className="py-3">
- <label className="block text-sm text-bambu-gray mb-2">{t('settings.apiKeyForTesting')}</label>
- <input
- type="text"
- value={testApiKey}
- onChange={(e) => setTestApiKey(e.target.value)}
- placeholder={t('settings.apiKeyPlaceholder')}
- 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"
- />
- <p className="text-xs text-bambu-gray mt-2">
- {t('settings.apiKeyHint')}
- </p>
- </CardContent>
- </Card>
- <APIBrowser apiKey={testApiKey} />
- </div>
- </div>
- )}
- {/* Virtual Printer Tab */}
- {activeTab === 'virtual-printer' && (
- <VirtualPrinterSettings />
- )}
- {/* Filament Tab */}
- {activeTab === 'filament' && localSettings && (
- <div className="flex flex-col lg:flex-row gap-6 lg:gap-8">
- {/* Left Column - AMS Display Thresholds */}
- <div className="flex-1 lg:max-w-xl">
- <Card>
- <CardHeader>
- <h2 className="text-lg font-semibold text-white">{t('settings.amsDisplayThresholds')}</h2>
- </CardHeader>
- <CardContent className="space-y-4">
- <p className="text-sm text-bambu-gray">
- {t('settings.amsThresholdsDescription')}
- </p>
- {/* Humidity Thresholds */}
- <div className="space-y-3">
- <div className="flex items-center gap-2 text-white">
- <Droplets className="w-4 h-4 text-blue-400" />
- <span className="font-medium">{t('settings.humidity')}</span>
- </div>
- <div className="grid grid-cols-2 gap-3">
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- {t('settings.goodGreen')} ≤
- </label>
- <div className="flex items-center gap-2">
- <input
- type="number"
- min="0"
- max="100"
- value={localSettings.ams_humidity_good ?? 40}
- onChange={(e) => updateSetting('ams_humidity_good', parseInt(e.target.value) || 40)}
- 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"
- />
- <span className="text-bambu-gray">%</span>
- </div>
- </div>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- {t('settings.fairOrange')} ≤
- </label>
- <div className="flex items-center gap-2">
- <input
- type="number"
- min="0"
- max="100"
- value={localSettings.ams_humidity_fair ?? 60}
- onChange={(e) => updateSetting('ams_humidity_fair', parseInt(e.target.value) || 60)}
- 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"
- />
- <span className="text-bambu-gray">%</span>
- </div>
- </div>
- </div>
- <p className="text-xs text-bambu-gray">
- {t('settings.aboveFairBad')}
- </p>
- </div>
- {/* Temperature Thresholds */}
- <div className="space-y-3 pt-2 border-t border-bambu-dark-tertiary">
- <div className="flex items-center gap-2 text-white">
- <Thermometer className="w-4 h-4 text-orange-400" />
- <span className="font-medium">{t('settings.temperature')}</span>
- </div>
- <div className="grid grid-cols-2 gap-3">
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- {t('settings.goodBlue')} ≤
- </label>
- <div className="flex items-center gap-2">
- <input
- type="number"
- step="0.5"
- min="0"
- max="60"
- value={localSettings.ams_temp_good ?? 28}
- onChange={(e) => updateSetting('ams_temp_good', parseFloat(e.target.value) || 28)}
- 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"
- />
- <span className="text-bambu-gray">°C</span>
- </div>
- </div>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- {t('settings.fairOrange')} ≤
- </label>
- <div className="flex items-center gap-2">
- <input
- type="number"
- step="0.5"
- min="0"
- max="60"
- value={localSettings.ams_temp_fair ?? 35}
- onChange={(e) => updateSetting('ams_temp_fair', parseFloat(e.target.value) || 35)}
- 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"
- />
- <span className="text-bambu-gray">°C</span>
- </div>
- </div>
- </div>
- <p className="text-xs text-bambu-gray">
- {t('settings.aboveFairHot')}
- </p>
- </div>
- {/* History Retention */}
- <div className="space-y-3 pt-4 border-t border-bambu-dark-tertiary">
- <div className="flex items-center gap-2 text-white">
- <Database className="w-4 h-4 text-purple-400" />
- <span className="font-medium">{t('settings.historyRetention')}</span>
- </div>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- {t('settings.keepSensorHistory')}
- </label>
- <div className="flex items-center gap-2">
- <input
- type="number"
- min="1"
- max="365"
- value={localSettings.ams_history_retention_days ?? 30}
- onChange={(e) => updateSetting('ams_history_retention_days', parseInt(e.target.value) || 30)}
- 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"
- />
- <span className="text-bambu-gray">{t('common.days')}</span>
- </div>
- </div>
- <p className="text-xs text-bambu-gray">
- {t('settings.historyRetentionDescription')}
- </p>
- </div>
- {/* Per-Printer Mapping Default */}
- <div className="space-y-3 pt-4 border-t border-bambu-dark-tertiary">
- <div className="flex items-center gap-2 text-white">
- <Printer className="w-4 h-4 text-bambu-green" />
- <span className="font-medium">{t('settings.printModal')}</span>
- </div>
- <div className="flex items-center justify-between">
- <div>
- <label className="block text-sm text-white">
- {t('settings.expandCustomMapping')}
- </label>
- <p className="text-xs text-bambu-gray mt-0.5">
- {t('settings.expandCustomMappingDescription')}
- </p>
- </div>
- <label className="relative inline-flex items-center cursor-pointer">
- <input
- type="checkbox"
- checked={localSettings.per_printer_mapping_expanded ?? false}
- onChange={(e) => updateSetting('per_printer_mapping_expanded', e.target.checked)}
- className="sr-only peer"
- />
- <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>
- </label>
- </div>
- </div>
- </CardContent>
- </Card>
- </div>
- {/* Right Column - Spoolman Integration */}
- <div className="flex-1 lg:max-w-xl">
- <SpoolmanSettings />
- </div>
- </div>
- )}
- {/* Delete API Key Confirmation */}
- {showDeleteAPIKeyConfirm !== null && (
- <ConfirmModal
- title={t('settings.deleteApiKeyTitle')}
- message={t('settings.deleteApiKeyMessage')}
- confirmText={t('settings.deleteKey')}
- variant="danger"
- onConfirm={() => {
- deleteAPIKeyMutation.mutate(showDeleteAPIKeyConfirm);
- setShowDeleteAPIKeyConfirm(null);
- }}
- onCancel={() => setShowDeleteAPIKeyConfirm(null)}
- />
- )}
- {/* Smart Plug Modal */}
- {showPlugModal && (
- <AddSmartPlugModal
- plug={editingPlug}
- onClose={() => {
- setShowPlugModal(false);
- setEditingPlug(null);
- }}
- />
- )}
- {/* Notification Modal */}
- {showNotificationModal && (
- <AddNotificationModal
- provider={editingProvider}
- onClose={() => {
- setShowNotificationModal(false);
- setEditingProvider(null);
- }}
- />
- )}
- {/* Template Editor Modal */}
- {editingTemplate && (
- <NotificationTemplateEditor
- template={editingTemplate}
- onClose={() => setEditingTemplate(null)}
- />
- )}
- {/* Notification Log Viewer */}
- {showLogViewer && (
- <NotificationLogViewer
- onClose={() => setShowLogViewer(false)}
- />
- )}
- {/* Confirm Modal: Clear Notification Logs */}
- {showClearLogsConfirm && (
- <ConfirmModal
- title={t('settings.clearNotificationLogs')}
- message={t('settings.clearLogsMessage')}
- confirmText={t('settings.clearLogs')}
- variant="warning"
- onConfirm={async () => {
- setShowClearLogsConfirm(false);
- try {
- const result = await api.clearNotificationLogs(30);
- showToast(result.message, 'success');
- } catch {
- showToast(t('settings.toast.clearLogsFailed'), 'error');
- }
- }}
- onCancel={() => setShowClearLogsConfirm(false)}
- />
- )}
- {/* Confirm Modal: Clear Local Storage */}
- {showClearStorageConfirm && (
- <ConfirmModal
- title={t('settings.resetUiPreferences')}
- message={t('settings.resetUiPreferencesMessage')}
- confirmText={t('settings.resetPreferences')}
- variant="default"
- onConfirm={() => {
- setShowClearStorageConfirm(false);
- localStorage.clear();
- showToast(t('settings.toast.uiPreferencesReset'), 'success');
- setTimeout(() => window.location.reload(), 1000);
- }}
- onCancel={() => setShowClearStorageConfirm(false)}
- />
- )}
- {/* Confirm Modal: Bulk Plug Action */}
- {showBulkPlugConfirm && (
- <ConfirmModal
- title={`Turn All Plugs ${showBulkPlugConfirm === 'on' ? 'On' : 'Off'}`}
- 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!' : ''}`}
- confirmText={`Turn All ${showBulkPlugConfirm === 'on' ? 'On' : 'Off'}`}
- variant={showBulkPlugConfirm === 'off' ? 'danger' : 'warning'}
- onConfirm={() => {
- const action = showBulkPlugConfirm;
- setShowBulkPlugConfirm(null);
- bulkPlugActionMutation.mutate(action);
- }}
- onCancel={() => setShowBulkPlugConfirm(null)}
- />
- )}
- {/* Release Notes Modal */}
- {showReleaseNotes && updateCheck?.release_notes && (
- <div
- className="fixed inset-0 bg-black/50 flex items-center justify-center z-50 p-4"
- onClick={() => setShowReleaseNotes(false)}
- >
- <Card className="w-full max-w-2xl max-h-[80vh] flex flex-col" onClick={(e: React.MouseEvent) => e.stopPropagation()}>
- <CardHeader className="flex flex-row items-center justify-between shrink-0">
- <div>
- <h2 className="text-lg font-semibold text-white">
- Release Notes - v{updateCheck.latest_version}
- </h2>
- {updateCheck.release_name && updateCheck.release_name !== updateCheck.latest_version && (
- <p className="text-sm text-bambu-gray">{updateCheck.release_name}</p>
- )}
- </div>
- <button
- onClick={() => setShowReleaseNotes(false)}
- className="p-1 rounded hover:bg-bambu-dark-tertiary text-bambu-gray hover:text-white"
- >
- <X className="w-5 h-5" />
- </button>
- </CardHeader>
- <CardContent className="overflow-y-auto flex-1">
- <pre className="text-sm text-bambu-gray whitespace-pre-wrap font-sans">
- {updateCheck.release_notes}
- </pre>
- </CardContent>
- <div className="p-4 border-t border-bambu-dark-tertiary shrink-0 flex gap-2">
- {updateCheck.release_url && (
- <a
- href={updateCheck.release_url}
- target="_blank"
- rel="noopener noreferrer"
- className="flex-1"
- >
- <Button variant="secondary" className="w-full">
- <ExternalLink className="w-4 h-4" />
- View on GitHub
- </Button>
- </a>
- )}
- <Button
- onClick={() => setShowReleaseNotes(false)}
- className="flex-1"
- >
- Close
- </Button>
- </div>
- </Card>
- </div>
- )}
- {/* Users Tab */}
- {activeTab === 'users' && (
- <div className="space-y-6">
- {/* Auth Toggle Header */}
- <Card>
- <CardContent className="py-4">
- <div className="flex items-center justify-between">
- <div className="flex items-center gap-3">
- <div className={`w-10 h-10 rounded-full flex items-center justify-center ${authEnabled ? 'bg-green-500/20' : 'bg-gray-500/20'}`}>
- {authEnabled ? (
- <Lock className="w-5 h-5 text-green-400" />
- ) : (
- <Unlock className="w-5 h-5 text-gray-400" />
- )}
- </div>
- <div>
- <h3 className="text-white font-medium">{t('settings.authentication')}</h3>
- <p className="text-sm text-bambu-gray">
- {authEnabled
- ? t('settings.authEnabledDescription')
- : t('settings.authDisabledDescription')}
- </p>
- </div>
- </div>
- {!authEnabled ? (
- <Button onClick={() => navigate('/setup')}>
- <Lock className="w-4 h-4" />
- {t('common.enable')}
- </Button>
- ) : user?.is_admin && (
- <Button variant="secondary" onClick={() => setShowDisableAuthConfirm(true)}>
- <Unlock className="w-4 h-4" />
- {t('common.disable')}
- </Button>
- )}
- </div>
- </CardContent>
- </Card>
- {authEnabled && (
- <div className="grid grid-cols-1 xl:grid-cols-2 gap-6">
- {/* Left Column: Current User + User List */}
- <div className="space-y-6">
- {/* Current User Card */}
- {user && (
- <Card>
- <CardHeader>
- <div className="flex items-center justify-between">
- <h3 className="text-lg font-semibold text-white flex items-center gap-2">
- <Users className="w-5 h-5 text-bambu-green" />
- {t('settings.currentUser')}
- </h3>
- <Button size="sm" variant="ghost" onClick={() => setShowChangePasswordModal(true)}>
- <Key className="w-4 h-4" />
- {t('settings.changePassword')}
- </Button>
- </div>
- </CardHeader>
- <CardContent>
- <div className="flex items-center justify-between">
- <div>
- <p className="text-white font-medium text-lg">{user.username}</p>
- <div className="flex flex-wrap gap-1 mt-2">
- {user.is_admin && (
- <span className="px-2 py-0.5 rounded-full text-xs font-medium bg-purple-500/20 text-purple-300">
- {t('settings.admin')}
- </span>
- )}
- {user.groups?.map(group => (
- <span
- key={group.id}
- className={`px-2 py-0.5 rounded-full text-xs font-medium ${
- group.name === 'Administrators'
- ? 'bg-purple-500/20 text-purple-300'
- : group.name === 'Operators'
- ? 'bg-blue-500/20 text-blue-300'
- : group.name === 'Viewers'
- ? 'bg-green-500/20 text-green-300'
- : 'bg-gray-500/20 text-gray-300'
- }`}
- >
- {group.name}
- </span>
- ))}
- </div>
- </div>
- </div>
- </CardContent>
- </Card>
- )}
- {/* User List */}
- <Card>
- <CardHeader>
- <div className="flex items-center justify-between">
- <h3 className="text-lg font-semibold text-white flex items-center gap-2">
- <Users className="w-5 h-5 text-bambu-green" />
- {t('settings.users')}
- </h3>
- {hasPermission('users:create') && (
- <Button
- size="sm"
- onClick={() => {
- setShowCreateUserModal(true);
- setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
- }}
- >
- <Plus className="w-4 h-4" />
- {t('settings.addUser')}
- </Button>
- )}
- </div>
- </CardHeader>
- <CardContent>
- {usersLoading ? (
- <div className="flex items-center justify-center py-8">
- <Loader2 className="w-6 h-6 text-bambu-green animate-spin" />
- </div>
- ) : usersData.length === 0 ? (
- <p className="text-center text-bambu-gray py-8">{t('settings.noUsersFound')}</p>
- ) : (
- <div className="divide-y divide-bambu-dark-tertiary">
- {usersData.map((userItem) => (
- <div key={userItem.id} className="py-3 flex items-center justify-between">
- <div className="flex-1 min-w-0">
- <p className="text-white font-medium truncate">{userItem.username}</p>
- <div className="flex flex-wrap gap-1 mt-1">
- {userItem.is_admin && (
- <span className="px-2 py-0.5 rounded-full text-xs font-medium bg-purple-500/20 text-purple-300">
- {t('settings.admin')}
- </span>
- )}
- {userItem.groups?.map(group => (
- <span
- key={group.id}
- className={`px-2 py-0.5 rounded-full text-xs font-medium ${
- group.name === 'Administrators'
- ? 'bg-purple-500/20 text-purple-300'
- : group.name === 'Operators'
- ? 'bg-blue-500/20 text-blue-300'
- : group.name === 'Viewers'
- ? 'bg-green-500/20 text-green-300'
- : 'bg-gray-500/20 text-gray-300'
- }`}
- >
- {group.name}
- </span>
- ))}
- </div>
- </div>
- <div className="flex items-center gap-1 ml-4">
- {hasPermission('users:update') && (
- <Button size="sm" variant="ghost" onClick={() => startEditUser(userItem)}>
- <Edit2 className="w-4 h-4" />
- </Button>
- )}
- {hasPermission('users:delete') && userItem.id !== user?.id && (
- <Button size="sm" variant="ghost" onClick={() => handleDeleteUserClick(userItem.id)}>
- <Trash2 className="w-4 h-4" />
- </Button>
- )}
- </div>
- </div>
- ))}
- </div>
- )}
- </CardContent>
- </Card>
- </div>
- {/* Right Column: Groups */}
- <div>
- <Card>
- <CardHeader>
- <div className="flex items-center justify-between">
- <h3 className="text-lg font-semibold text-white flex items-center gap-2">
- <Shield className="w-5 h-5 text-bambu-green" />
- {t('settings.groups')}
- </h3>
- {hasPermission('groups:create') && (
- <Button
- size="sm"
- onClick={() => {
- setShowCreateGroupModal(true);
- resetGroupForm();
- }}
- >
- <Plus className="w-4 h-4" />
- {t('settings.addGroup')}
- </Button>
- )}
- </div>
- </CardHeader>
- <CardContent>
- {groupsLoading ? (
- <div className="flex items-center justify-center py-8">
- <Loader2 className="w-6 h-6 text-bambu-green animate-spin" />
- </div>
- ) : groupsData.length === 0 ? (
- <p className="text-center text-bambu-gray py-8">{t('settings.noGroupsFound')}</p>
- ) : (
- <div className="divide-y divide-bambu-dark-tertiary">
- {groupsData.map((group) => (
- <div key={group.id} className="py-3">
- <div className="flex items-center justify-between">
- <div className="flex items-center gap-2">
- <Shield
- className={`w-4 h-4 ${
- group.name === 'Administrators'
- ? 'text-purple-400'
- : group.name === 'Operators'
- ? 'text-blue-400'
- : group.name === 'Viewers'
- ? 'text-green-400'
- : 'text-bambu-gray'
- }`}
- />
- <span className="text-white font-medium">{group.name}</span>
- {group.is_system && (
- <span className="px-2 py-0.5 rounded text-xs bg-yellow-500/20 text-yellow-400">
- {t('settings.system')}
- </span>
- )}
- </div>
- <div className="flex items-center gap-1">
- {hasPermission('groups:update') && (
- <Button size="sm" variant="ghost" onClick={() => startEditGroup(group)}>
- <Edit2 className="w-4 h-4" />
- </Button>
- )}
- {hasPermission('groups:delete') && !group.is_system && (
- <Button size="sm" variant="ghost" onClick={() => setDeleteGroupId(group.id)}>
- <Trash2 className="w-4 h-4" />
- </Button>
- )}
- </div>
- </div>
- <p className="text-sm text-bambu-gray mt-1 ml-6">
- {group.description || t('settings.noDescription')}
- </p>
- <div className="flex items-center gap-4 mt-2 ml-6 text-xs text-bambu-gray">
- <span>{t('settings.userCount', { count: group.user_count })}</span>
- <span>{t('settings.permissionCount', { count: group.permissions.length })}</span>
- </div>
- </div>
- ))}
- </div>
- )}
- </CardContent>
- </Card>
- </div>
- </div>
- )}
- {/* Auth Disabled Info */}
- {!authEnabled && (
- <Card>
- <CardContent className="py-6">
- <div className="text-center">
- <Unlock className="w-12 h-12 text-bambu-gray mx-auto mb-4" />
- <h3 className="text-lg font-medium text-white mb-2">{t('settings.authDisabledTitle')}</h3>
- <p className="text-sm text-bambu-gray mb-4 max-w-md mx-auto">
- {t('settings.authDisabledMessage')}
- </p>
- <ul className="space-y-2 text-sm text-bambu-gray mb-6 text-left max-w-xs mx-auto">
- <li className="flex items-start gap-2">
- <CheckCircle className="w-4 h-4 text-bambu-green mt-0.5 flex-shrink-0" />
- <span>{t('settings.authDisabledFeature1')}</span>
- </li>
- <li className="flex items-start gap-2">
- <CheckCircle className="w-4 h-4 text-bambu-green mt-0.5 flex-shrink-0" />
- <span>{t('settings.authDisabledFeature2')}</span>
- </li>
- <li className="flex items-start gap-2">
- <CheckCircle className="w-4 h-4 text-bambu-green mt-0.5 flex-shrink-0" />
- <span>{t('settings.authDisabledFeature3')}</span>
- </li>
- </ul>
- <Button onClick={() => navigate('/setup')}>
- <Lock className="w-4 h-4" />
- {t('settings.enableAuthentication')}
- </Button>
- </div>
- </CardContent>
- </Card>
- )}
- </div>
- )}
- {/* Create User Modal */}
- {showCreateUserModal && (
- <div
- className="fixed inset-0 bg-black flex items-center justify-center z-50 p-4"
- onClick={() => {
- setShowCreateUserModal(false);
- setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
- }}
- >
- <Card
- className="w-full max-w-md"
- onClick={(e: React.MouseEvent) => e.stopPropagation()}
- >
- <CardHeader>
- <div className="flex items-center justify-between">
- <div className="flex items-center gap-2">
- <Users className="w-5 h-5 text-bambu-green" />
- <h2 className="text-lg font-semibold text-white">{t('settings.createUser')}</h2>
- </div>
- <Button
- variant="ghost"
- size="sm"
- onClick={() => {
- setShowCreateUserModal(false);
- setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
- }}
- >
- <X className="w-5 h-5" />
- </Button>
- </div>
- </CardHeader>
- <CardContent>
- <div className="space-y-4">
- <div>
- <label className="block text-sm font-medium text-white mb-2">{t('settings.username')}</label>
- <input
- type="text"
- value={userFormData.username}
- onChange={(e) => setUserFormData({ ...userFormData, username: e.target.value })}
- 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"
- placeholder={t('settings.enterUsername')}
- autoComplete="username"
- />
- </div>
- <div>
- <label className="block text-sm font-medium text-white mb-2">{t('settings.password')}</label>
- <input
- type="password"
- value={userFormData.password}
- onChange={(e) => setUserFormData({ ...userFormData, password: e.target.value })}
- 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"
- placeholder={t('settings.enterPassword')}
- autoComplete="new-password"
- minLength={6}
- />
- </div>
- <div>
- <label className="block text-sm font-medium text-white mb-2">{t('settings.confirmPassword')}</label>
- <input
- type="password"
- value={userFormData.confirmPassword}
- onChange={(e) => setUserFormData({ ...userFormData, confirmPassword: e.target.value })}
- 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 ${
- userFormData.confirmPassword && userFormData.password !== userFormData.confirmPassword
- ? 'border-red-500'
- : 'border-bambu-dark-tertiary'
- }`}
- placeholder={t('settings.confirmPasswordPlaceholder')}
- autoComplete="new-password"
- minLength={6}
- />
- {userFormData.confirmPassword && userFormData.password !== userFormData.confirmPassword && (
- <p className="text-red-400 text-xs mt-1">{t('settings.passwordsDoNotMatch')}</p>
- )}
- </div>
- <div>
- <label className="block text-sm font-medium text-white mb-2">Groups</label>
- <div className="space-y-2 max-h-40 overflow-y-auto p-2 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg">
- {groupsData.map(group => (
- <label
- key={group.id}
- className="flex items-center gap-3 px-2 py-1.5 rounded hover:bg-bambu-dark-tertiary cursor-pointer"
- >
- <input
- type="checkbox"
- checked={userFormData.group_ids.includes(group.id)}
- onChange={() => toggleUserGroup(group.id)}
- className="w-4 h-4 rounded border-bambu-gray text-bambu-green focus:ring-bambu-green focus:ring-offset-0 bg-bambu-dark"
- />
- <span className="text-sm text-white">{group.name}</span>
- {group.is_system && (
- <span className="text-xs text-yellow-400">(System)</span>
- )}
- </label>
- ))}
- {groupsData.length === 0 && (
- <p className="text-sm text-bambu-gray">{t('settings.noGroupsAvailable')}</p>
- )}
- </div>
- </div>
- </div>
- <div className="mt-6 flex justify-end gap-3">
- <Button
- variant="secondary"
- onClick={() => {
- setShowCreateUserModal(false);
- setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
- }}
- >
- Cancel
- </Button>
- <Button
- onClick={handleCreateUser}
- disabled={createUserMutation.isPending || !userFormData.username || !userFormData.password || userFormData.password !== userFormData.confirmPassword || userFormData.password.length < 6}
- >
- {createUserMutation.isPending ? (
- <>
- <Loader2 className="w-4 h-4 animate-spin" />
- Creating...
- </>
- ) : (
- <>
- <Plus className="w-4 h-4" />
- Create User
- </>
- )}
- </Button>
- </div>
- </CardContent>
- </Card>
- </div>
- )}
- {/* Edit User Modal */}
- {showEditUserModal && editingUserId !== null && (
- <div
- className="fixed inset-0 bg-black flex items-center justify-center z-50 p-4"
- onClick={() => {
- setShowEditUserModal(false);
- setEditingUserId(null);
- setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
- }}
- >
- <Card
- className="w-full max-w-md"
- onClick={(e: React.MouseEvent) => e.stopPropagation()}
- >
- <CardHeader>
- <div className="flex items-center justify-between">
- <div className="flex items-center gap-2">
- <Edit2 className="w-5 h-5 text-bambu-green" />
- <h2 className="text-lg font-semibold text-white">{t('settings.editUser')}</h2>
- </div>
- <Button
- variant="ghost"
- size="sm"
- onClick={() => {
- setShowEditUserModal(false);
- setEditingUserId(null);
- setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
- }}
- >
- <X className="w-5 h-5" />
- </Button>
- </div>
- </CardHeader>
- <CardContent>
- <div className="space-y-4">
- <div>
- <label className="block text-sm font-medium text-white mb-2">{t('settings.username')}</label>
- <input
- type="text"
- value={userFormData.username}
- onChange={(e) => setUserFormData({ ...userFormData, username: e.target.value })}
- 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"
- placeholder={t('settings.enterUsername')}
- autoComplete="username"
- />
- </div>
- <div>
- <label className="block text-sm font-medium text-white mb-2">
- Password <span className="text-bambu-gray font-normal">(leave blank to keep current)</span>
- </label>
- <input
- type="password"
- value={userFormData.password}
- onChange={(e) => setUserFormData({ ...userFormData, password: e.target.value, confirmPassword: '' })}
- 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"
- placeholder={t('settings.enterNewPassword')}
- autoComplete="new-password"
- minLength={6}
- />
- </div>
- {userFormData.password && (
- <div>
- <label className="block text-sm font-medium text-white mb-2">{t('settings.confirmPassword')}</label>
- <input
- type="password"
- value={userFormData.confirmPassword}
- onChange={(e) => setUserFormData({ ...userFormData, confirmPassword: e.target.value })}
- 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 ${
- userFormData.confirmPassword && userFormData.password !== userFormData.confirmPassword
- ? 'border-red-500'
- : 'border-bambu-dark-tertiary'
- }`}
- placeholder={t('settings.confirmNewPassword')}
- autoComplete="new-password"
- minLength={6}
- />
- {userFormData.confirmPassword && userFormData.password !== userFormData.confirmPassword && (
- <p className="text-red-400 text-xs mt-1">{t('settings.passwordsDoNotMatch')}</p>
- )}
- </div>
- )}
- <div>
- <label className="block text-sm font-medium text-white mb-2">Groups</label>
- <div className="space-y-2 max-h-40 overflow-y-auto p-2 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg">
- {groupsData.map(group => (
- <label
- key={group.id}
- className="flex items-center gap-3 px-2 py-1.5 rounded hover:bg-bambu-dark-tertiary cursor-pointer"
- >
- <input
- type="checkbox"
- checked={userFormData.group_ids.includes(group.id)}
- onChange={() => toggleUserGroup(group.id)}
- className="w-4 h-4 rounded border-bambu-gray text-bambu-green focus:ring-bambu-green focus:ring-offset-0 bg-bambu-dark"
- />
- <span className="text-sm text-white">{group.name}</span>
- {group.is_system && (
- <span className="text-xs text-yellow-400">(System)</span>
- )}
- </label>
- ))}
- </div>
- </div>
- </div>
- <div className="mt-6 flex justify-end gap-3">
- <Button
- variant="secondary"
- onClick={() => {
- setShowEditUserModal(false);
- setEditingUserId(null);
- setUserFormData({ username: '', password: '', confirmPassword: '', role: 'user', group_ids: [] });
- }}
- >
- Cancel
- </Button>
- <Button
- onClick={() => handleUpdateUser(editingUserId)}
- disabled={updateUserMutation.isPending || !userFormData.username || !!(userFormData.password && (userFormData.password !== userFormData.confirmPassword || userFormData.password.length < 6))}
- >
- {updateUserMutation.isPending ? (
- <>
- <Loader2 className="w-4 h-4 animate-spin" />
- Saving...
- </>
- ) : (
- <>
- <Save className="w-4 h-4" />
- Save Changes
- </>
- )}
- </Button>
- </div>
- </CardContent>
- </Card>
- </div>
- )}
- {/* Delete User Confirmation Modal */}
- {deleteUserId !== null && (
- <div
- className="fixed inset-0 bg-black/80 flex items-center justify-center z-50 p-4"
- onClick={() => {
- setDeleteUserId(null);
- setDeleteUserItemCounts(null);
- }}
- >
- <Card
- className="w-full max-w-md"
- onClick={(e: React.MouseEvent) => e.stopPropagation()}
- >
- <CardHeader>
- <div className="flex items-center gap-2 text-red-400">
- <Trash2 className="w-5 h-5" />
- <h3 className="text-lg font-semibold">{t('settings.deleteUserTitle')}</h3>
- </div>
- </CardHeader>
- <CardContent className="space-y-4">
- {deleteUserLoading ? (
- <div className="flex items-center justify-center py-4">
- <div className="animate-spin rounded-full h-6 w-6 border-2 border-bambu-green border-t-transparent" />
- </div>
- ) : deleteUserItemCounts && (deleteUserItemCounts.archives + deleteUserItemCounts.queue_items + deleteUserItemCounts.library_files > 0) ? (
- <>
- <p className="text-white">{t('settings.userHasCreated')}</p>
- <ul className="list-disc list-inside text-bambu-gray space-y-1">
- {deleteUserItemCounts.archives > 0 && (
- <li>{deleteUserItemCounts.archives} archive{deleteUserItemCounts.archives !== 1 ? 's' : ''}</li>
- )}
- {deleteUserItemCounts.queue_items > 0 && (
- <li>{deleteUserItemCounts.queue_items} queue item{deleteUserItemCounts.queue_items !== 1 ? 's' : ''}</li>
- )}
- {deleteUserItemCounts.library_files > 0 && (
- <li>{deleteUserItemCounts.library_files} library file{deleteUserItemCounts.library_files !== 1 ? 's' : ''}</li>
- )}
- </ul>
- <p className="text-bambu-gray text-sm">{t('settings.userItemsQuestion')}</p>
- <div className="flex flex-col gap-2">
- <Button
- variant="danger"
- onClick={() => deleteUserMutation.mutate({ id: deleteUserId, deleteItems: true })}
- disabled={deleteUserMutation.isPending}
- className="justify-center"
- >
- Delete user AND their items
- </Button>
- <Button
- variant="secondary"
- onClick={() => deleteUserMutation.mutate({ id: deleteUserId, deleteItems: false })}
- disabled={deleteUserMutation.isPending}
- className="justify-center"
- >
- Delete user, keep items (become ownerless)
- </Button>
- <Button
- variant="ghost"
- onClick={() => {
- setDeleteUserId(null);
- setDeleteUserItemCounts(null);
- }}
- disabled={deleteUserMutation.isPending}
- className="justify-center"
- >
- Cancel
- </Button>
- </div>
- </>
- ) : (
- <>
- <p className="text-white">{t('settings.deleteUserConfirm')}</p>
- <p className="text-bambu-gray text-sm">{t('settings.actionCannotBeUndone')}</p>
- <div className="flex gap-2 justify-end">
- <Button
- variant="ghost"
- onClick={() => {
- setDeleteUserId(null);
- setDeleteUserItemCounts(null);
- }}
- disabled={deleteUserMutation.isPending}
- >
- Cancel
- </Button>
- <Button
- variant="danger"
- onClick={() => deleteUserMutation.mutate({ id: deleteUserId, deleteItems: false })}
- disabled={deleteUserMutation.isPending}
- >
- Delete User
- </Button>
- </div>
- </>
- )}
- </CardContent>
- </Card>
- </div>
- )}
- {/* Create/Edit Group Modal */}
- {(showCreateGroupModal || editingGroup) && (
- <div
- className="fixed inset-0 bg-black flex items-center justify-center z-50 p-4"
- onClick={() => {
- setShowCreateGroupModal(false);
- setEditingGroup(null);
- resetGroupForm();
- }}
- >
- <Card
- className="w-full max-w-2xl max-h-[90vh] overflow-y-auto"
- onClick={(e: React.MouseEvent) => e.stopPropagation()}
- >
- <CardHeader>
- <div className="flex items-center justify-between">
- <div className="flex items-center gap-2">
- <Shield className="w-5 h-5 text-bambu-green" />
- <h2 className="text-lg font-semibold text-white">
- {editingGroup ? 'Edit Group' : 'Create Group'}
- </h2>
- </div>
- <Button
- variant="ghost"
- size="sm"
- onClick={() => {
- setShowCreateGroupModal(false);
- setEditingGroup(null);
- resetGroupForm();
- }}
- >
- <X className="w-5 h-5" />
- </Button>
- </div>
- </CardHeader>
- <CardContent>
- <div className="space-y-4">
- <div>
- <label className="block text-sm font-medium text-white mb-2">{t('settings.groupName')}</label>
- <input
- type="text"
- value={groupFormData.name}
- onChange={(e) => setGroupFormData({ ...groupFormData, name: e.target.value })}
- disabled={editingGroup?.is_system}
- 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"
- placeholder={t('settings.enterGroupName')}
- />
- {editingGroup?.is_system && (
- <p className="text-xs text-yellow-400 mt-1">{t('settings.systemGroupWarning')}</p>
- )}
- </div>
- <div>
- <label className="block text-sm font-medium text-white mb-2">Description</label>
- <textarea
- value={groupFormData.description}
- onChange={(e) => setGroupFormData({ ...groupFormData, description: e.target.value })}
- rows={2}
- 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"
- placeholder={t('settings.enterDescriptionOptional')}
- />
- </div>
- <div>
- <label className="block text-sm font-medium text-white mb-2">
- Permissions ({groupFormData.permissions.length} selected)
- </label>
- <div className="space-y-2 max-h-96 overflow-y-auto">
- {permissionsData?.categories.map((category) => (
- <div key={category.name} className="border border-bambu-dark-tertiary rounded-lg overflow-hidden">
- <div
- className="flex items-center justify-between px-4 py-2 bg-bambu-dark-secondary cursor-pointer hover:bg-bambu-dark-tertiary transition-colors"
- onClick={() => toggleCategory(category.name)}
- >
- <div className="flex items-center gap-3">
- <button
- type="button"
- onClick={(e) => {
- e.stopPropagation();
- toggleCategoryPermissions(category, !isCategoryFullySelected(category));
- }}
- className={`w-5 h-5 rounded border flex items-center justify-center transition-colors ${
- isCategoryFullySelected(category)
- ? 'bg-bambu-green border-bambu-green'
- : isCategoryPartiallySelected(category)
- ? 'bg-bambu-green/50 border-bambu-green'
- : 'border-bambu-gray hover:border-white'
- }`}
- >
- {(isCategoryFullySelected(category) || isCategoryPartiallySelected(category)) && (
- <Check className="w-3 h-3 text-white" />
- )}
- </button>
- <span className="text-white font-medium">{category.name}</span>
- <span className="text-xs text-bambu-gray">
- ({category.permissions.filter((p) => groupFormData.permissions.includes(p.value)).length}/
- {category.permissions.length})
- </span>
- </div>
- {expandedCategories.has(category.name) ? (
- <ChevronDown className="w-4 h-4 text-bambu-gray" />
- ) : (
- <ChevronRight className="w-4 h-4 text-bambu-gray" />
- )}
- </div>
- {expandedCategories.has(category.name) && (
- <div className="p-3 bg-bambu-dark space-y-2">
- {category.permissions.map((perm) => (
- <label
- key={perm.value}
- className="flex items-center gap-3 px-2 py-1.5 rounded hover:bg-bambu-dark-secondary cursor-pointer"
- >
- <input
- type="checkbox"
- checked={groupFormData.permissions.includes(perm.value)}
- onChange={() => togglePermission(perm.value)}
- className="w-4 h-4 rounded border-bambu-gray text-bambu-green focus:ring-bambu-green focus:ring-offset-0 bg-bambu-dark-secondary"
- />
- <span className="text-sm text-bambu-gray">{perm.label}</span>
- </label>
- ))}
- </div>
- )}
- </div>
- ))}
- </div>
- </div>
- </div>
- <div className="mt-6 flex justify-end gap-3">
- <Button
- variant="secondary"
- onClick={() => {
- setShowCreateGroupModal(false);
- setEditingGroup(null);
- resetGroupForm();
- }}
- >
- Cancel
- </Button>
- <Button
- onClick={editingGroup ? handleUpdateGroup : handleCreateGroup}
- disabled={createGroupMutation.isPending || updateGroupMutation.isPending || !groupFormData.name.trim()}
- >
- {(createGroupMutation.isPending || updateGroupMutation.isPending) ? (
- <>
- <Loader2 className="w-4 h-4 animate-spin" />
- {editingGroup ? 'Saving...' : 'Creating...'}
- </>
- ) : (
- <>
- <Save className="w-4 h-4" />
- {editingGroup ? 'Save Changes' : 'Create Group'}
- </>
- )}
- </Button>
- </div>
- </CardContent>
- </Card>
- </div>
- )}
- {/* Delete Group Confirmation Modal */}
- {deleteGroupId !== null && (
- <ConfirmModal
- title={t('settings.deleteGroupTitle')}
- message={t('settings.deleteGroupMessage')}
- confirmText={t('settings.deleteGroup')}
- variant="danger"
- onConfirm={() => {
- deleteGroupMutation.mutate(deleteGroupId);
- setDeleteGroupId(null);
- }}
- onCancel={() => setDeleteGroupId(null)}
- />
- )}
- {/* Backup Tab */}
- {activeTab === 'backup' && (
- <GitHubBackupSettings />
- )}
- {/* Disable Authentication Confirmation Modal */}
- {showDisableAuthConfirm && (
- <ConfirmModal
- title={t('settings.disableAuthenticationTitle')}
- message={t('settings.disableAuthenticationMessage')}
- confirmText={t('settings.disableAuthentication')}
- variant="danger"
- onConfirm={async () => {
- try {
- await api.disableAuth();
- showToast(t('settings.toast.authDisabled'), 'success');
- await refreshAuth();
- setShowDisableAuthConfirm(false);
- // Refresh the page to ensure all protected routes are accessible
- window.location.href = '/';
- } catch (error: unknown) {
- const message = error instanceof Error ? error.message : t('settings.toast.authDisableFailed');
- showToast(message, 'error');
- }
- }}
- onCancel={() => setShowDisableAuthConfirm(false)}
- />
- )}
- {/* Change Password Modal */}
- {showChangePasswordModal && (
- <div
- className="fixed inset-0 bg-black/70 flex items-center justify-center z-50 p-4"
- onClick={() => {
- setShowChangePasswordModal(false);
- setChangePasswordData({ currentPassword: '', newPassword: '', confirmPassword: '' });
- }}
- >
- <Card
- className="w-full max-w-md"
- onClick={(e: React.MouseEvent) => e.stopPropagation()}
- >
- <CardHeader>
- <div className="flex items-center justify-between">
- <div className="flex items-center gap-2">
- <Key className="w-5 h-5 text-bambu-green" />
- <h2 className="text-lg font-semibold text-white">{t('settings.changePassword')}</h2>
- </div>
- <Button
- variant="ghost"
- size="sm"
- onClick={() => {
- setShowChangePasswordModal(false);
- setChangePasswordData({ currentPassword: '', newPassword: '', confirmPassword: '' });
- }}
- >
- <X className="w-5 h-5" />
- </Button>
- </div>
- </CardHeader>
- <CardContent>
- <div className="space-y-4">
- <div>
- <label className="block text-sm font-medium text-white mb-2">
- Current Password
- </label>
- <input
- type="password"
- value={changePasswordData.currentPassword}
- onChange={(e) => setChangePasswordData({ ...changePasswordData, currentPassword: e.target.value })}
- 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"
- placeholder={t('settings.enterCurrentPassword')}
- autoComplete="current-password"
- />
- </div>
- <div>
- <label className="block text-sm font-medium text-white mb-2">
- New Password
- </label>
- <input
- type="password"
- value={changePasswordData.newPassword}
- onChange={(e) => setChangePasswordData({ ...changePasswordData, newPassword: e.target.value })}
- 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"
- placeholder={t('settings.enterNewPasswordMin6')}
- autoComplete="new-password"
- minLength={6}
- />
- </div>
- <div>
- <label className="block text-sm font-medium text-white mb-2">
- Confirm New Password
- </label>
- <input
- type="password"
- value={changePasswordData.confirmPassword}
- onChange={(e) => setChangePasswordData({ ...changePasswordData, confirmPassword: e.target.value })}
- 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 ${
- changePasswordData.confirmPassword && changePasswordData.newPassword !== changePasswordData.confirmPassword
- ? 'border-red-500'
- : 'border-bambu-dark-tertiary'
- }`}
- placeholder={t('settings.confirmNewPassword')}
- autoComplete="new-password"
- minLength={6}
- />
- {changePasswordData.confirmPassword && changePasswordData.newPassword !== changePasswordData.confirmPassword && (
- <p className="text-red-400 text-xs mt-1">{t('settings.passwordsDoNotMatch')}</p>
- )}
- </div>
- </div>
- <div className="mt-6 flex justify-end gap-3">
- <Button
- variant="secondary"
- onClick={() => {
- setShowChangePasswordModal(false);
- setChangePasswordData({ currentPassword: '', newPassword: '', confirmPassword: '' });
- }}
- >
- Cancel
- </Button>
- <Button
- onClick={async () => {
- if (changePasswordData.newPassword !== changePasswordData.confirmPassword) {
- showToast(t('settings.toast.passwordsDoNotMatch'), 'error');
- return;
- }
- if (changePasswordData.newPassword.length < 6) {
- showToast(t('settings.toast.passwordTooShort'), 'error');
- return;
- }
- setChangePasswordLoading(true);
- try {
- await api.changePassword(changePasswordData.currentPassword, changePasswordData.newPassword);
- showToast(t('settings.toast.passwordChanged'), 'success');
- setShowChangePasswordModal(false);
- setChangePasswordData({ currentPassword: '', newPassword: '', confirmPassword: '' });
- } catch (error: unknown) {
- const message = error instanceof Error ? error.message : 'Failed to change password';
- showToast(message, 'error');
- } finally {
- setChangePasswordLoading(false);
- }
- }}
- disabled={changePasswordLoading || !changePasswordData.currentPassword || !changePasswordData.newPassword || changePasswordData.newPassword !== changePasswordData.confirmPassword || changePasswordData.newPassword.length < 6}
- >
- {changePasswordLoading ? (
- <>
- <Loader2 className="w-4 h-4 animate-spin" />
- Changing...
- </>
- ) : (
- <>
- <Key className="w-4 h-4" />
- Change Password
- </>
- )}
- </Button>
- </div>
- </CardContent>
- </Card>
- </div>
- )}
- </div>
- );
- }
|