PrintersPage.tsx 367 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810481148124813481448154816481748184819482048214822482348244825482648274828482948304831483248334834483548364837483848394840484148424843484448454846484748484849485048514852485348544855485648574858485948604861486248634864486548664867486848694870487148724873487448754876487748784879488048814882488348844885488648874888488948904891489248934894489548964897489848994900490149024903490449054906490749084909491049114912491349144915491649174918491949204921492249234924492549264927492849294930493149324933493449354936493749384939494049414942494349444945494649474948494949504951495249534954495549564957495849594960496149624963496449654966496749684969497049714972497349744975497649774978497949804981498249834984498549864987498849894990499149924993499449954996499749984999500050015002500350045005500650075008500950105011501250135014501550165017501850195020502150225023502450255026502750285029503050315032503350345035503650375038503950405041504250435044504550465047504850495050505150525053505450555056505750585059506050615062506350645065506650675068506950705071507250735074507550765077507850795080508150825083508450855086508750885089509050915092509350945095509650975098509951005101510251035104510551065107510851095110511151125113511451155116511751185119512051215122512351245125512651275128512951305131513251335134513551365137513851395140514151425143514451455146514751485149515051515152515351545155515651575158515951605161516251635164516551665167516851695170517151725173517451755176517751785179518051815182518351845185518651875188518951905191519251935194519551965197519851995200520152025203520452055206520752085209521052115212521352145215521652175218521952205221522252235224522552265227522852295230523152325233523452355236523752385239524052415242524352445245524652475248524952505251525252535254525552565257525852595260526152625263526452655266526752685269527052715272527352745275527652775278527952805281528252835284528552865287528852895290529152925293529452955296529752985299530053015302530353045305530653075308530953105311531253135314531553165317531853195320532153225323532453255326532753285329533053315332533353345335533653375338533953405341534253435344534553465347534853495350535153525353535453555356535753585359536053615362536353645365536653675368536953705371537253735374537553765377537853795380538153825383538453855386538753885389539053915392539353945395539653975398539954005401540254035404540554065407540854095410541154125413541454155416541754185419542054215422542354245425542654275428542954305431543254335434543554365437543854395440544154425443544454455446544754485449545054515452545354545455545654575458545954605461546254635464546554665467546854695470547154725473547454755476547754785479548054815482548354845485548654875488548954905491549254935494549554965497549854995500550155025503550455055506550755085509551055115512551355145515551655175518551955205521552255235524552555265527552855295530553155325533553455355536553755385539554055415542554355445545554655475548554955505551555255535554555555565557555855595560556155625563556455655566556755685569557055715572557355745575557655775578557955805581558255835584558555865587558855895590559155925593559455955596559755985599560056015602560356045605560656075608560956105611561256135614561556165617561856195620562156225623562456255626562756285629563056315632563356345635563656375638563956405641564256435644564556465647564856495650565156525653565456555656565756585659566056615662566356645665566656675668566956705671567256735674567556765677567856795680568156825683568456855686568756885689569056915692569356945695569656975698569957005701570257035704570557065707570857095710571157125713571457155716571757185719572057215722572357245725572657275728572957305731573257335734573557365737573857395740574157425743574457455746574757485749575057515752575357545755575657575758575957605761576257635764576557665767576857695770577157725773577457755776577757785779578057815782578357845785578657875788578957905791579257935794579557965797579857995800580158025803580458055806580758085809581058115812581358145815581658175818581958205821582258235824582558265827582858295830583158325833583458355836583758385839584058415842584358445845584658475848584958505851585258535854585558565857585858595860586158625863586458655866586758685869587058715872587358745875587658775878587958805881588258835884588558865887588858895890589158925893589458955896589758985899590059015902590359045905590659075908590959105911591259135914591559165917591859195920592159225923592459255926592759285929593059315932593359345935593659375938593959405941594259435944594559465947594859495950595159525953595459555956595759585959596059615962596359645965596659675968596959705971597259735974597559765977597859795980598159825983598459855986598759885989599059915992599359945995599659975998599960006001600260036004600560066007600860096010601160126013601460156016601760186019602060216022602360246025602660276028602960306031603260336034603560366037603860396040604160426043604460456046604760486049605060516052605360546055605660576058605960606061606260636064606560666067606860696070607160726073607460756076607760786079608060816082608360846085608660876088608960906091609260936094609560966097609860996100610161026103610461056106610761086109611061116112611361146115611661176118611961206121612261236124612561266127612861296130613161326133613461356136613761386139614061416142614361446145614661476148614961506151615261536154615561566157615861596160616161626163616461656166616761686169617061716172617361746175617661776178617961806181618261836184618561866187618861896190619161926193619461956196619761986199620062016202620362046205620662076208620962106211621262136214621562166217621862196220622162226223622462256226622762286229623062316232623362346235623662376238623962406241624262436244624562466247624862496250625162526253625462556256625762586259626062616262626362646265626662676268626962706271627262736274627562766277627862796280628162826283628462856286628762886289629062916292629362946295629662976298629963006301630263036304630563066307630863096310631163126313631463156316631763186319632063216322632363246325632663276328632963306331633263336334633563366337633863396340634163426343634463456346634763486349635063516352635363546355635663576358635963606361636263636364636563666367636863696370637163726373637463756376637763786379638063816382638363846385638663876388638963906391639263936394639563966397639863996400640164026403640464056406640764086409641064116412641364146415641664176418641964206421642264236424642564266427642864296430643164326433643464356436643764386439644064416442644364446445644664476448644964506451645264536454645564566457645864596460646164626463646464656466646764686469647064716472647364746475647664776478647964806481648264836484648564866487648864896490649164926493649464956496649764986499650065016502650365046505650665076508650965106511651265136514651565166517651865196520652165226523652465256526652765286529653065316532653365346535653665376538653965406541654265436544654565466547654865496550655165526553655465556556655765586559656065616562656365646565656665676568656965706571657265736574657565766577657865796580658165826583658465856586658765886589659065916592659365946595659665976598659966006601660266036604660566066607660866096610661166126613661466156616661766186619662066216622662366246625662666276628662966306631663266336634663566366637663866396640664166426643664466456646664766486649665066516652665366546655665666576658665966606661666266636664666566666667666866696670667166726673667466756676667766786679668066816682668366846685668666876688668966906691669266936694669566966697669866996700670167026703670467056706670767086709671067116712671367146715671667176718671967206721672267236724672567266727672867296730673167326733673467356736673767386739674067416742674367446745674667476748674967506751675267536754675567566757675867596760676167626763676467656766676767686769677067716772677367746775677667776778677967806781678267836784678567866787678867896790679167926793679467956796679767986799680068016802680368046805680668076808680968106811681268136814681568166817681868196820682168226823682468256826682768286829683068316832683368346835683668376838683968406841684268436844684568466847684868496850685168526853685468556856685768586859686068616862686368646865686668676868686968706871687268736874687568766877687868796880688168826883688468856886688768886889689068916892689368946895689668976898689969006901690269036904690569066907690869096910691169126913691469156916691769186919692069216922692369246925692669276928692969306931693269336934693569366937693869396940694169426943694469456946694769486949695069516952695369546955695669576958695969606961696269636964696569666967696869696970697169726973697469756976697769786979698069816982698369846985698669876988698969906991699269936994699569966997699869997000700170027003700470057006700770087009701070117012701370147015701670177018701970207021702270237024702570267027702870297030703170327033703470357036703770387039704070417042704370447045704670477048704970507051705270537054705570567057705870597060706170627063706470657066706770687069707070717072707370747075707670777078707970807081708270837084708570867087708870897090709170927093709470957096709770987099710071017102710371047105710671077108710971107111711271137114711571167117711871197120712171227123712471257126712771287129713071317132713371347135713671377138713971407141714271437144714571467147714871497150715171527153715471557156715771587159716071617162716371647165716671677168716971707171717271737174717571767177717871797180718171827183718471857186718771887189719071917192719371947195719671977198719972007201720272037204720572067207720872097210721172127213721472157216721772187219722072217222722372247225722672277228722972307231723272337234723572367237723872397240724172427243724472457246724772487249725072517252725372547255725672577258725972607261726272637264726572667267726872697270727172727273727472757276727772787279728072817282728372847285728672877288728972907291729272937294729572967297729872997300730173027303730473057306730773087309731073117312731373147315731673177318731973207321732273237324732573267327732873297330733173327333733473357336733773387339734073417342734373447345734673477348734973507351735273537354735573567357735873597360736173627363736473657366736773687369737073717372737373747375737673777378737973807381738273837384738573867387738873897390739173927393739473957396739773987399740074017402740374047405740674077408740974107411741274137414741574167417741874197420742174227423742474257426742774287429743074317432743374347435743674377438743974407441744274437444744574467447744874497450745174527453745474557456745774587459746074617462746374647465746674677468746974707471747274737474747574767477747874797480748174827483748474857486748774887489749074917492749374947495749674977498749975007501750275037504750575067507750875097510751175127513751475157516751775187519752075217522752375247525752675277528752975307531753275337534753575367537753875397540754175427543754475457546754775487549755075517552755375547555755675577558755975607561756275637564756575667567756875697570757175727573757475757576757775787579758075817582758375847585758675877588758975907591759275937594759575967597759875997600760176027603760476057606760776087609761076117612761376147615761676177618761976207621762276237624762576267627762876297630763176327633763476357636763776387639764076417642764376447645764676477648764976507651765276537654765576567657765876597660766176627663766476657666766776687669767076717672767376747675767676777678767976807681768276837684768576867687768876897690
  1. import { useState, useEffect, useLayoutEffect, useMemo, useRef, useCallback } from 'react';
  2. import { compareFwVersions } from '../utils/firmwareVersion';
  3. import { formatPrintName } from '../utils/printName';
  4. import { computePopoverPosition } from '../utils/popoverPosition';
  5. // AMS drying popover dimensions — w-[240px] on the popover, estimated height
  6. // covers header + filament select + temp slider + duration + rotate-tray
  7. // toggle + buttons. Over-estimating is fine (flip-above kicks in slightly
  8. // earlier); under-estimating leaves the popover clipped off the bottom (the
  9. // original bug at #1447).
  10. const DRYING_POPOVER_WIDTH = 240;
  11. const DRYING_POPOVER_ESTIMATED_HEIGHT = 320;
  12. import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
  13. import { useTranslation } from 'react-i18next';
  14. import { useTheme } from '../contexts/ThemeContext';
  15. import { useAuth } from '../contexts/AuthContext';
  16. import {
  17. Plus,
  18. Link,
  19. Unlink,
  20. Signal,
  21. Clock,
  22. MoreVertical,
  23. Trash2,
  24. RefreshCw,
  25. RotateCw,
  26. Box,
  27. HardDrive,
  28. AlertTriangle,
  29. AlertCircle,
  30. Terminal,
  31. Power,
  32. PowerOff,
  33. Zap,
  34. Wrench,
  35. ChevronDown,
  36. Filter,
  37. Pencil,
  38. ArrowUp,
  39. ArrowDown,
  40. Layers,
  41. Video,
  42. Search,
  43. Loader2,
  44. Square,
  45. Pause,
  46. Play,
  47. X,
  48. Fan,
  49. Wind,
  50. AirVent,
  51. Download,
  52. ScanSearch,
  53. CheckCircle,
  54. CheckSquare,
  55. XCircle,
  56. User,
  57. Home,
  58. Printer as PrinterIcon,
  59. Info,
  60. Cable,
  61. Flame,
  62. Snowflake,
  63. Gauge,
  64. DoorOpen,
  65. DoorClosed,
  66. MoveVertical,
  67. LogIn,
  68. LogOut,
  69. MoreHorizontal,
  70. SlidersHorizontal,
  71. Stethoscope,
  72. } from 'lucide-react';
  73. import { useNavigate } from 'react-router-dom';
  74. import { api, discoveryApi, firmwareApi, withStreamToken, ApiError } from '../api/client';
  75. import { formatDateOnly, formatETA, formatDuration, parseUTCDate } from '../utils/date';
  76. import type { Printer, PrinterCreate, PrinterStatus, AMSUnit, DiscoveredPrinter, FirmwareUpdateInfo, FirmwareUploadStatus, LinkedSpoolInfo, SpoolAssignment, HMSError, InventorySpool, SmartPlug, PrinterDiagnosticResult } from '../api/client';
  77. import { Card, CardContent } from '../components/Card';
  78. import { Button } from '../components/Button';
  79. import { ConfirmModal } from '../components/ConfirmModal';
  80. import { BulkPrinterToolbar, type PrinterState } from '../components/BulkPrinterToolbar';
  81. import { FileManagerModal } from '../components/FileManagerModal';
  82. import { EmbeddedCameraViewer } from '../components/EmbeddedCameraViewer';
  83. import { MQTTDebugModal } from '../components/MQTTDebugModal';
  84. import { HMSErrorModal, filterKnownHMSErrors } from '../components/HMSErrorModal';
  85. import { PrinterQueueWidget } from '../components/PrinterQueueWidget';
  86. import { AMSHistoryModal } from '../components/AMSHistoryModal';
  87. import { FilamentHoverCard, EmptySlotHoverCard } from '../components/FilamentHoverCard';
  88. import { LinkSpoolModal } from '../components/LinkSpoolModal';
  89. import { AssignSpoolModal } from '../components/AssignSpoolModal';
  90. import { ConfigureAmsSlotModal } from '../components/ConfigureAmsSlotModal';
  91. import { useToast } from '../contexts/ToastContext';
  92. import { ChamberLight } from '../components/icons/ChamberLight';
  93. import { PlateClearedIcon } from '../components/icons/PlateClearedIcon';
  94. import { SkipObjectsModal, SkipObjectsIcon } from '../components/SkipObjectsModal';
  95. import { FileUploadModal } from '../components/FileUploadModal';
  96. import { PrintModal } from '../components/PrintModal';
  97. import { PrinterInfoModal } from '../components/PrinterInfoModal';
  98. import { getGlobalTrayId, getFillBarColor, getSpoolmanFillLevel, getFallbackSpoolTag, isBambuLabSpool } from '../utils/amsHelpers';
  99. import { getPrinterImage, getWifiStrength, filterCompatibleQueueItems } from '../utils/printer';
  100. import { FilamentSlotCircle } from '../components/FilamentSlotCircle';
  101. import { Collapsible } from '../components/Collapsible';
  102. import { ConnectionDiagnosticModal, DiagnosticChecklist } from '../components/ConnectionDiagnostic';
  103. import { getColorName, parseFilamentColor, isLightColor } from '../utils/colors';
  104. export interface SpoolmanSlotAssignmentRow {
  105. printer_id: number;
  106. ams_id: number;
  107. tray_id: number;
  108. spoolman_spool_id: number;
  109. }
  110. // Color names resolve via getColorName() which reads the backend color_catalog
  111. // (loaded once by ColorCatalogProvider). No hardcoded tables here — see #857.
  112. // Format K value with 3 decimal places, default to 0.020 if null
  113. function formatKValue(k: number | null | undefined): string {
  114. const value = k ?? 0.020;
  115. return value.toFixed(3);
  116. }
  117. // Nozzle side indicators (Bambu Lab style - square badge with L/R)
  118. function NozzleBadge({ side }: { side: 'L' | 'R' }) {
  119. const { mode } = useTheme();
  120. // Light mode: #e7f5e9 (light green), Dark mode: #1a4d2e (dark green)
  121. const bgColor = mode === 'dark' ? '#1a4d2e' : '#e7f5e9';
  122. return (
  123. <span
  124. className="inline-flex items-center justify-center w-4 h-4 text-[10px] font-bold rounded"
  125. style={{ backgroundColor: bgColor, color: '#00ae42' }}
  126. >
  127. {side}
  128. </span>
  129. );
  130. }
  131. // Expand nozzle type codes to material names
  132. // Handles full text ("hardened_steel"), 2-char codes ("HS"/"HH"), and 4-char codes ("HS01")
  133. // Material mapping: 00=stainless steel, 01=hardened steel, 05=tungsten carbide
  134. function nozzleTypeName(type: string, t: (key: string) => string): string {
  135. if (!type) return '';
  136. // Full text names (from main nozzle info)
  137. if (type.includes('hardened')) return t('printers.nozzleHardenedSteel');
  138. if (type.includes('stainless')) return t('printers.nozzleStainlessSteel');
  139. if (type.includes('tungsten')) return t('printers.nozzleTungstenCarbide');
  140. // 4-char codes (e.g. "HS01"): last 2 digits = material
  141. if (type.length >= 4) {
  142. const material = type.slice(2, 4);
  143. if (material === '00') return t('printers.nozzleStainlessSteel');
  144. if (material === '01') return t('printers.nozzleHardenedSteel');
  145. if (material === '05') return t('printers.nozzleTungstenCarbide');
  146. }
  147. // 2-digit numeric codes
  148. if (type === '00') return t('printers.nozzleStainlessSteel');
  149. if (type === '01') return t('printers.nozzleHardenedSteel');
  150. if (type === '05') return t('printers.nozzleTungstenCarbide');
  151. // 2-char alpha codes: H prefix = hardened steel
  152. if (type.startsWith('H')) return t('printers.nozzleHardenedSteel');
  153. return type;
  154. }
  155. // Parse flow type from nozzle type code
  156. // HH = high flow, HS = standard/normal
  157. function nozzleFlowName(type: string, t: (key: string) => string): string {
  158. if (!type) return '';
  159. if (type.startsWith('HH')) return t('printers.nozzleHighFlow');
  160. if (type.startsWith('HS')) return t('printers.nozzleStandardFlow');
  161. return '';
  162. }
  163. // Per-slot hover card for nozzle rack
  164. // activeStatus: when true, show "Active" instead of "Mounted"/"Docked" (for hotend nozzles)
  165. function NozzleSlotHoverCard({ slot, index, activeStatus, filamentName, children }: {
  166. slot: import('../api/client').NozzleRackSlot;
  167. index: number;
  168. activeStatus?: boolean;
  169. filamentName?: string;
  170. children: React.ReactNode;
  171. }) {
  172. const { t } = useTranslation();
  173. const [isVisible, setIsVisible] = useState(false);
  174. const [position, setPosition] = useState<'top' | 'bottom'>('top');
  175. const triggerRef = useRef<HTMLDivElement>(null);
  176. const cardRef = useRef<HTMLDivElement>(null);
  177. const timeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null);
  178. const isEmpty = !slot.nozzle_diameter && !slot.nozzle_type;
  179. const isMounted = slot.stat === 1;
  180. useEffect(() => {
  181. if (isVisible && triggerRef.current && cardRef.current) {
  182. const triggerRect = triggerRef.current.getBoundingClientRect();
  183. const cardHeight = cardRef.current.offsetHeight;
  184. const headerHeight = 56;
  185. const spaceAbove = triggerRect.top - headerHeight;
  186. const spaceBelow = window.innerHeight - triggerRect.bottom;
  187. if (spaceAbove < cardHeight + 12 && spaceBelow > spaceAbove) {
  188. setPosition('bottom');
  189. } else {
  190. setPosition('top');
  191. }
  192. }
  193. }, [isVisible]);
  194. const handleMouseEnter = () => {
  195. if (timeoutRef.current) clearTimeout(timeoutRef.current);
  196. timeoutRef.current = setTimeout(() => setIsVisible(true), 80);
  197. };
  198. const handleMouseLeave = () => {
  199. if (timeoutRef.current) clearTimeout(timeoutRef.current);
  200. timeoutRef.current = setTimeout(() => setIsVisible(false), 100);
  201. };
  202. useEffect(() => {
  203. return () => {
  204. if (timeoutRef.current) clearTimeout(timeoutRef.current);
  205. };
  206. }, []);
  207. const filamentCss = parseFilamentColor(slot.filament_color);
  208. const typeFull = nozzleTypeName(slot.nozzle_type, t);
  209. const flowFull = nozzleFlowName(slot.nozzle_type, t);
  210. return (
  211. <div
  212. ref={triggerRef}
  213. className="relative"
  214. onMouseEnter={handleMouseEnter}
  215. onMouseLeave={handleMouseLeave}
  216. >
  217. {children}
  218. {isVisible && (
  219. <div
  220. ref={cardRef}
  221. className={`
  222. absolute left-1/2 -translate-x-1/2 z-50
  223. ${position === 'top' ? 'bottom-full mb-2' : 'top-full mt-2'}
  224. animate-in fade-in-0 zoom-in-95 duration-150
  225. `}
  226. style={{ maxWidth: 'calc(100vw - 24px)' }}
  227. >
  228. <div className="w-44 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg shadow-xl overflow-hidden backdrop-blur-sm">
  229. {isEmpty ? (
  230. <div className="px-3 py-2 text-xs text-bambu-gray text-center whitespace-nowrap">
  231. Slot {index + 1} — Empty
  232. </div>
  233. ) : (
  234. <div className="p-2.5 space-y-1.5">
  235. {/* Diameter */}
  236. <div className="flex items-center justify-between">
  237. <span className="text-[10px] uppercase tracking-wider text-bambu-gray font-medium">{t('printers.nozzleDiameter')}</span>
  238. <span className="text-xs text-white font-semibold">{slot.nozzle_diameter} mm</span>
  239. </div>
  240. {/* Type */}
  241. {typeFull && (
  242. <div className="flex items-center justify-between">
  243. <span className="text-[10px] uppercase tracking-wider text-bambu-gray font-medium">{t('printers.nozzleType')}</span>
  244. <span className="text-xs text-white font-semibold truncate max-w-[100px]">{typeFull}</span>
  245. </div>
  246. )}
  247. {/* Flow (hide if empty) */}
  248. {flowFull && (
  249. <div className="flex items-center justify-between">
  250. <span className="text-[10px] uppercase tracking-wider text-bambu-gray font-medium">{t('printers.nozzleFlow')}</span>
  251. <span className="text-xs text-white font-semibold">{flowFull}</span>
  252. </div>
  253. )}
  254. {/* Status badge */}
  255. <div className="flex items-center justify-between">
  256. <span className="text-[10px] uppercase tracking-wider text-bambu-gray font-medium">{t('printers.nozzleStatus')}</span>
  257. <span className={`text-[10px] font-bold px-1.5 py-0.5 rounded ${
  258. activeStatus || isMounted
  259. ? 'bg-green-900/50 text-green-400'
  260. : 'bg-bambu-dark-tertiary text-bambu-gray'
  261. }`}>
  262. {activeStatus ? t('printers.nozzleActive') : isMounted ? t('printers.nozzleMounted') : t('printers.nozzleDocked')}
  263. </span>
  264. </div>
  265. {/* Wear (hide if null) */}
  266. {slot.wear != null && (
  267. <div className="flex items-center justify-between">
  268. <span className="text-[10px] uppercase tracking-wider text-bambu-gray font-medium">{t('printers.nozzleWear')}</span>
  269. <span className="text-xs text-white font-semibold">{slot.wear}%</span>
  270. </div>
  271. )}
  272. {/* Max Temp (hide if 0) */}
  273. {slot.max_temp > 0 && (
  274. <div className="flex items-center justify-between">
  275. <span className="text-[10px] uppercase tracking-wider text-bambu-gray font-medium">{t('printers.nozzleMaxTemp')}</span>
  276. <span className="text-xs text-white font-semibold">{slot.max_temp}°C</span>
  277. </div>
  278. )}
  279. {/* Serial (hide if empty) */}
  280. {slot.serial_number && (
  281. <div className="flex items-center justify-between">
  282. <span className="text-[10px] uppercase tracking-wider text-bambu-gray font-medium">{t('printers.nozzleSerial')}</span>
  283. <span className="text-[10px] text-white font-mono truncate max-w-[80px]">{slot.serial_number}</span>
  284. </div>
  285. )}
  286. {/* Filament: material type + color swatch (hide if no color) */}
  287. {(filamentCss || slot.filament_type) && (
  288. <div className="flex items-center justify-between">
  289. <span className="text-[10px] uppercase tracking-wider text-bambu-gray font-medium">{t('printers.nozzleFilament')}</span>
  290. <div className="flex items-center gap-1">
  291. {filamentCss && (
  292. <div className="w-3 h-3 rounded-sm border border-white/20" style={{ backgroundColor: filamentCss }} />
  293. )}
  294. <span className="text-[10px] text-white font-semibold truncate max-w-[100px]">{filamentName || slot.filament_type || slot.filament_id || ''}</span>
  295. </div>
  296. </div>
  297. )}
  298. </div>
  299. )}
  300. </div>
  301. {/* Arrow pointer */}
  302. <div
  303. className={`
  304. absolute left-1/2 -translate-x-1/2 w-0 h-0
  305. border-l-[6px] border-l-transparent
  306. border-r-[6px] border-r-transparent
  307. ${position === 'top'
  308. ? 'top-full border-t-[6px] border-t-bambu-dark-tertiary'
  309. : 'bottom-full border-b-[6px] border-b-bambu-dark-tertiary'}
  310. `}
  311. />
  312. </div>
  313. )}
  314. </div>
  315. );
  316. }
  317. // Dual-nozzle hover card showing L and R nozzle details side by side
  318. function DualNozzleHoverCard({ leftSlot, rightSlot, activeNozzle, filamentInfo, children }: {
  319. leftSlot?: import('../api/client').NozzleRackSlot;
  320. rightSlot?: import('../api/client').NozzleRackSlot;
  321. activeNozzle: 'L' | 'R';
  322. filamentInfo?: Record<string, { name: string; k: number | null }>;
  323. children: React.ReactNode;
  324. }) {
  325. const { t } = useTranslation();
  326. const [isVisible, setIsVisible] = useState(false);
  327. const [position, setPosition] = useState<'top' | 'bottom'>('top');
  328. const triggerRef = useRef<HTMLDivElement>(null);
  329. const cardRef = useRef<HTMLDivElement>(null);
  330. const timeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null);
  331. useEffect(() => {
  332. if (isVisible && triggerRef.current && cardRef.current) {
  333. const triggerRect = triggerRef.current.getBoundingClientRect();
  334. const cardHeight = cardRef.current.offsetHeight;
  335. const headerHeight = 56;
  336. const spaceAbove = triggerRect.top - headerHeight;
  337. const spaceBelow = window.innerHeight - triggerRect.bottom;
  338. if (spaceAbove < cardHeight + 12 && spaceBelow > spaceAbove) {
  339. setPosition('bottom');
  340. } else {
  341. setPosition('top');
  342. }
  343. }
  344. }, [isVisible]);
  345. const handleMouseEnter = () => {
  346. if (timeoutRef.current) clearTimeout(timeoutRef.current);
  347. timeoutRef.current = setTimeout(() => setIsVisible(true), 80);
  348. };
  349. const handleMouseLeave = () => {
  350. if (timeoutRef.current) clearTimeout(timeoutRef.current);
  351. timeoutRef.current = setTimeout(() => setIsVisible(false), 100);
  352. };
  353. useEffect(() => {
  354. return () => { if (timeoutRef.current) clearTimeout(timeoutRef.current); };
  355. }, []);
  356. if (!leftSlot && !rightSlot) return <>{children}</>;
  357. const renderColumn = (slot: import('../api/client').NozzleRackSlot, side: 'L' | 'R') => {
  358. const isActive = activeNozzle === side;
  359. const typeFull = nozzleTypeName(slot.nozzle_type, t);
  360. const flowFull = nozzleFlowName(slot.nozzle_type, t);
  361. const filamentCss = parseFilamentColor(slot.filament_color);
  362. const filamentName = slot.filament_id ? filamentInfo?.[slot.filament_id]?.name : undefined;
  363. return (
  364. <div className="flex-1 space-y-1.5">
  365. <div className={`text-[10px] font-bold pb-1 border-b border-bambu-dark-tertiary/50 ${isActive ? 'text-amber-400' : 'text-bambu-gray'}`}>
  366. {side === 'L' ? t('common.left') : t('common.right')}
  367. </div>
  368. {slot.nozzle_diameter && (
  369. <div className="flex items-center justify-between">
  370. <span className="text-[10px] text-bambu-gray">{t('printers.nozzleDiameter')}</span>
  371. <span className="text-xs text-white font-semibold">{slot.nozzle_diameter} mm</span>
  372. </div>
  373. )}
  374. {typeFull && (
  375. <div className="flex items-center justify-between">
  376. <span className="text-[10px] text-bambu-gray">{t('printers.nozzleType')}</span>
  377. <span className="text-[10px] text-white font-semibold">{typeFull}</span>
  378. </div>
  379. )}
  380. {flowFull && (
  381. <div className="flex items-center justify-between">
  382. <span className="text-[10px] text-bambu-gray">{t('printers.nozzleFlow')}</span>
  383. <span className="text-[10px] text-white font-semibold">{flowFull}</span>
  384. </div>
  385. )}
  386. <div className="flex items-center justify-between">
  387. <span className="text-[10px] text-bambu-gray">{t('printers.nozzleStatus')}</span>
  388. <span className={`text-[10px] font-bold px-1.5 py-0.5 rounded ${
  389. isActive
  390. ? 'bg-green-900/50 text-green-400'
  391. : 'bg-bambu-dark-tertiary text-bambu-gray'
  392. }`}>
  393. {isActive ? t('printers.nozzleActive') : t('printers.nozzleIdle')}
  394. </span>
  395. </div>
  396. {slot.wear != null && (
  397. <div className="flex items-center justify-between">
  398. <span className="text-[10px] text-bambu-gray">{t('printers.nozzleWear')}</span>
  399. <span className="text-xs text-white font-semibold">{slot.wear}%</span>
  400. </div>
  401. )}
  402. {/* Serial and max temp only available on the right (removable) nozzle */}
  403. {side === 'R' && slot.max_temp > 0 && (
  404. <div className="flex items-center justify-between">
  405. <span className="text-[10px] text-bambu-gray">{t('printers.nozzleMaxTemp')}</span>
  406. <span className="text-xs text-white font-semibold">{slot.max_temp}°C</span>
  407. </div>
  408. )}
  409. {side === 'R' && slot.serial_number && (
  410. <div className="flex items-center justify-between">
  411. <span className="text-[10px] text-bambu-gray">{t('printers.nozzleSerial')}</span>
  412. <span className="text-[10px] text-white font-mono">{slot.serial_number}</span>
  413. </div>
  414. )}
  415. {(filamentCss || slot.filament_type || slot.filament_id) && (
  416. <div className="flex items-center justify-between">
  417. <span className="text-[10px] text-bambu-gray">{t('printers.nozzleFilament')}</span>
  418. <div className="flex items-center gap-1">
  419. {filamentCss && (
  420. <div className="w-3 h-3 rounded-sm border border-white/20" style={{ backgroundColor: filamentCss }} />
  421. )}
  422. <span className="text-[10px] text-white font-semibold truncate max-w-[100px]">
  423. {filamentName || slot.filament_type || slot.filament_id || ''}
  424. </span>
  425. </div>
  426. </div>
  427. )}
  428. </div>
  429. );
  430. };
  431. return (
  432. <div
  433. ref={triggerRef}
  434. className="relative flex-1"
  435. onMouseEnter={handleMouseEnter}
  436. onMouseLeave={handleMouseLeave}
  437. >
  438. {children}
  439. {isVisible && (
  440. <div
  441. ref={cardRef}
  442. className={`
  443. absolute left-1/2 -translate-x-1/2 z-50
  444. ${position === 'top' ? 'bottom-full mb-2' : 'top-full mt-2'}
  445. animate-in fade-in-0 zoom-in-95 duration-150
  446. `}
  447. style={{ maxWidth: 'calc(100vw - 24px)' }}
  448. >
  449. <div className="w-96 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg shadow-xl overflow-hidden backdrop-blur-sm">
  450. <div className="p-2.5 flex gap-3">
  451. {leftSlot && renderColumn(leftSlot, 'L')}
  452. {leftSlot && rightSlot && <div className="w-px bg-bambu-dark-tertiary/50" />}
  453. {rightSlot && renderColumn(rightSlot, 'R')}
  454. </div>
  455. </div>
  456. {/* Arrow pointer */}
  457. <div
  458. className={`
  459. absolute left-1/2 -translate-x-1/2 w-0 h-0
  460. border-l-[6px] border-l-transparent
  461. border-r-[6px] border-r-transparent
  462. ${position === 'top'
  463. ? 'top-full border-t-[6px] border-t-bambu-dark-tertiary'
  464. : 'bottom-full border-b-[6px] border-b-bambu-dark-tertiary'}
  465. `}
  466. />
  467. </div>
  468. )}
  469. </div>
  470. );
  471. }
  472. // H2C Nozzle Rack Card — compact single row showing 6-position tool-changer dock
  473. function NozzleRackCard({ slots, filamentInfo }: { slots: import('../api/client').NozzleRackSlot[]; filamentInfo?: Record<string, { name: string; k: number | null }> }) {
  474. const { t } = useTranslation();
  475. // Rack nozzles only (IDs >= 2) — excludes L/R hotend nozzles (IDs 0, 1).
  476. // H2C rack slot IDs are fixed at 16..21. When a nozzle is picked up into the
  477. // hotend the firmware omits that rack ID entirely, so we must map by the fixed
  478. // base — computing it from min(present IDs) shifts everything left when slot 16
  479. // is the one currently mounted (#943).
  480. const rackNozzles = slots.filter(s => s.id >= 2);
  481. const RACK_SIZE = 6;
  482. const RACK_BASE_ID = 16;
  483. const rackSlots: (import('../api/client').NozzleRackSlot)[] = Array.from(
  484. { length: RACK_SIZE },
  485. (_, i) => rackNozzles.find(s => s.id === RACK_BASE_ID + i) ?? {
  486. id: -(i + 1), nozzle_type: '', nozzle_diameter: '', wear: null, stat: null,
  487. max_temp: 0, serial_number: '', filament_color: '', filament_id: '', filament_type: '',
  488. },
  489. );
  490. return (
  491. <div className="text-center px-2.5 py-1.5 bg-bambu-dark rounded-lg flex-[2_1_190px] flex flex-col justify-center">
  492. <p className="text-[9px] text-bambu-gray mb-1">{t('printers.nozzleRack')}</p>
  493. <div className="flex gap-[3px] justify-center">
  494. {rackSlots.map((slot, i) => {
  495. const isEmpty = !slot.nozzle_diameter && !slot.nozzle_type;
  496. const filamentBg = !isEmpty ? parseFilamentColor(slot.filament_color) : null;
  497. const lightBg = filamentBg ? isLightColor(slot.filament_color) : false;
  498. return (
  499. <NozzleSlotHoverCard key={slot.id >= 0 ? slot.id : `empty-${i}`} slot={slot} index={i} filamentName={slot.filament_id ? filamentInfo?.[slot.filament_id]?.name : undefined}>
  500. <div
  501. className={`w-7 h-7 rounded flex items-center justify-center cursor-default transition-colors border-b-2 ${
  502. isEmpty
  503. ? 'bg-bambu-dark-tertiary/20 border-bambu-dark-tertiary/20'
  504. : 'bg-bambu-dark-tertiary/40 border-bambu-dark-tertiary/40'
  505. }`}
  506. style={filamentBg ? { backgroundColor: filamentBg } : undefined}
  507. >
  508. <span className={`text-[10px] font-semibold ${isEmpty ? 'text-bambu-gray/30' : lightBg ? 'text-black/80' : 'text-white'}`}
  509. style={filamentBg && !lightBg ? { textShadow: '0 1px 3px rgba(0,0,0,0.9)' } : undefined}
  510. >
  511. {isEmpty ? '—' : (slot.nozzle_diameter || '?')}
  512. </span>
  513. </div>
  514. </NozzleSlotHoverCard>
  515. );
  516. })}
  517. </div>
  518. </div>
  519. );
  520. }
  521. // Water drop SVG - empty outline (Bambu Lab style from bambu-humidity)
  522. function WaterDropEmpty({ className }: { className?: string }) {
  523. return (
  524. <svg className={className} viewBox="0 0 36 54" fill="none" xmlns="http://www.w3.org/2000/svg">
  525. <path d="M17.8131 0.00538C18.4463 -0.15091 20.3648 3.14642 20.8264 3.84781C25.4187 10.816 35.3089 26.9368 35.9383 34.8694C37.4182 53.5822 11.882 61.3357 2.53721 45.3789C-1.73471 38.0791 0.016 32.2049 3.178 25.0232C6.99221 16.3662 12.6411 7.90372 17.8131 0.00538ZM18.3738 7.24807L17.5881 7.48441C14.4452 12.9431 10.917 18.2341 8.19369 23.9368C4.6808 31.29 1.18317 38.5479 7.69403 45.5657C17.3058 55.9228 34.9847 46.8808 31.4604 32.8681C29.2558 24.0969 22.4207 15.2913 18.3776 7.24807H18.3738Z" fill="#C3C2C1"/>
  526. </svg>
  527. );
  528. }
  529. // Water drop SVG - half filled with blue water (Bambu Lab style from bambu-humidity)
  530. function WaterDropHalf({ className }: { className?: string }) {
  531. return (
  532. <svg className={className} viewBox="0 0 35 53" fill="none" xmlns="http://www.w3.org/2000/svg">
  533. <path d="M17.3165 0.0038C17.932 -0.14959 19.7971 3.08645 20.2458 3.77481C24.7103 10.6135 34.3251 26.4346 34.937 34.2198C36.3757 52.5848 11.5505 60.1942 2.46584 44.534C-1.68714 37.3735 0.0148 31.6085 3.08879 24.5603C6.79681 16.0605 12.2884 7.75907 17.3165 0.0038ZM17.8615 7.11561L17.0977 7.34755C14.0423 12.7048 10.6124 17.8974 7.96483 23.4941C4.54975 30.7107 1.14949 37.8337 7.47908 44.721C16.8233 54.8856 34.01 46.0117 30.5838 32.2595C28.4405 23.6512 21.7957 15.0093 17.8652 7.11561H17.8615Z" fill="#C3C2C1"/>
  534. <path d="M5.03547 30.112C9.64453 30.4936 11.632 35.7985 16.4154 35.791C19.6339 35.7873 20.2161 33.2283 22.3853 31.6197C31.6776 24.7286 33.5835 37.4894 27.9881 44.4254C18.1878 56.5653 -1.16063 44.6013 5.03917 30.1158L5.03547 30.112Z" fill="#1F8FEB"/>
  535. </svg>
  536. );
  537. }
  538. // Water drop SVG - fully filled with blue water (Bambu Lab style from bambu-humidity)
  539. function WaterDropFull({ className }: { className?: string }) {
  540. return (
  541. <svg className={className} viewBox="0 0 36 54" fill="none" xmlns="http://www.w3.org/2000/svg">
  542. <path d="M17.9625 4.48059L4.77216 26.3154L2.08228 40.2175L10.0224 50.8414H23.1594L33.3246 42.1693V30.2455L17.9625 4.48059Z" fill="#1F8FEB"/>
  543. <path d="M17.7948 0.00538C18.4273 -0.15091 20.3438 3.14642 20.8048 3.84781C25.3921 10.816 35.2715 26.9368 35.9001 34.8694C37.3784 53.5822 11.8702 61.3357 2.53562 45.3789C-1.73163 38.0829 0.0134 32.2087 3.1757 25.027C6.98574 16.3662 12.6284 7.90372 17.7948 0.00538ZM18.3549 7.24807L17.57 7.48441C14.4306 12.9431 10.9063 18.2341 8.1859 23.9368C4.67686 31.29 1.18305 38.5479 7.68679 45.5657C17.2881 55.9228 34.9476 46.8808 31.4271 32.8681C29.2249 24.0969 22.3974 15.2913 18.3587 7.24807H18.3549Z" fill="#C3C2C1"/>
  544. </svg>
  545. );
  546. }
  547. // Thermometer SVG - empty outline
  548. function ThermometerEmpty({ className }: { className?: string }) {
  549. return (
  550. <svg className={className} viewBox="0 0 12 20" fill="none" xmlns="http://www.w3.org/2000/svg">
  551. <path d="M6 0.5C4.6 0.5 3.5 1.6 3.5 3V12.1C2.6 12.8 2 13.9 2 15C2 17.2 3.8 19 6 19C8.2 19 10 17.2 10 15C10 13.9 9.4 12.8 8.5 12.1V3C8.5 1.6 7.4 0.5 6 0.5Z" stroke="#C3C2C1" strokeWidth="1" fill="none"/>
  552. <circle cx="6" cy="15" r="2.5" stroke="#C3C2C1" strokeWidth="1" fill="none"/>
  553. </svg>
  554. );
  555. }
  556. // Thermometer SVG - half filled (gold - same as humidity fair)
  557. function ThermometerHalf({ className }: { className?: string }) {
  558. return (
  559. <svg className={className} viewBox="0 0 12 20" fill="none" xmlns="http://www.w3.org/2000/svg">
  560. <rect x="4.5" y="8" width="3" height="4.5" fill="#d4a017" rx="0.5"/>
  561. <circle cx="6" cy="15" r="2" fill="#d4a017"/>
  562. <path d="M6 0.5C4.6 0.5 3.5 1.6 3.5 3V12.1C2.6 12.8 2 13.9 2 15C2 17.2 3.8 19 6 19C8.2 19 10 17.2 10 15C10 13.9 9.4 12.8 8.5 12.1V3C8.5 1.6 7.4 0.5 6 0.5Z" stroke="#C3C2C1" strokeWidth="1" fill="none"/>
  563. </svg>
  564. );
  565. }
  566. // Thermometer SVG - fully filled (red - same as humidity bad)
  567. function ThermometerFull({ className }: { className?: string }) {
  568. return (
  569. <svg className={className} viewBox="0 0 12 20" fill="none" xmlns="http://www.w3.org/2000/svg">
  570. <rect x="4.5" y="3" width="3" height="9.5" fill="#c62828" rx="0.5"/>
  571. <circle cx="6" cy="15" r="2" fill="#c62828"/>
  572. <path d="M6 0.5C4.6 0.5 3.5 1.6 3.5 3V12.1C2.6 12.8 2 13.9 2 15C2 17.2 3.8 19 6 19C8.2 19 10 17.2 10 15C10 13.9 9.4 12.8 8.5 12.1V3C8.5 1.6 7.4 0.5 6 0.5Z" stroke="#C3C2C1" strokeWidth="1" fill="none"/>
  573. </svg>
  574. );
  575. }
  576. // Nozzle icon - schematic hot-end view (filament body + heater block + tip).
  577. // Added for visual parity with the thermometer icons on the dual-nozzle card
  578. // that previously had no icon at all (#1115, design by @m4rtini2).
  579. function NozzleIcon({ className }: { className?: string }) {
  580. return (
  581. <svg
  582. className={className}
  583. viewBox="0 0 24 24"
  584. fill="none"
  585. xmlns="http://www.w3.org/2000/svg"
  586. stroke="currentColor"
  587. strokeWidth="1.5"
  588. strokeLinecap="round"
  589. strokeLinejoin="round"
  590. >
  591. <rect x="9.2" y="3.4" width="5.6" height="8.1" />
  592. <rect x="6" y="11.5" width="12.1" height="3.7" />
  593. <path d="M 7.3 15.2 L 12.1 19.6 L 16.7 15.2" />
  594. </svg>
  595. );
  596. }
  597. // Heater thermometer icon - filled when heating, outline when off
  598. interface HeaterThermometerProps {
  599. className?: string;
  600. color: string; // The color class (e.g., "text-orange-400")
  601. isHeating: boolean;
  602. }
  603. function HeaterThermometer({ className, color, isHeating }: HeaterThermometerProps) {
  604. // Extract the actual color from Tailwind class for SVG fill
  605. const colorMap: Record<string, string> = {
  606. 'text-orange-400': '#fb923c',
  607. 'text-blue-400': '#60a5fa',
  608. 'text-green-400': '#4ade80',
  609. };
  610. const fillColor = colorMap[color] || '#888';
  611. // Glow style when heating
  612. const glowStyle = isHeating ? {
  613. filter: `drop-shadow(0 0 4px ${fillColor}) drop-shadow(0 0 8px ${fillColor})`,
  614. } : {};
  615. if (isHeating) {
  616. // Filled thermometer with glow - heater is ON
  617. return (
  618. <svg className={className} style={glowStyle} viewBox="0 0 12 20" fill="none" xmlns="http://www.w3.org/2000/svg">
  619. <rect x="4.5" y="3" width="3" height="9.5" fill={fillColor} rx="0.5"/>
  620. <circle cx="6" cy="15" r="2" fill={fillColor}/>
  621. <path d="M6 0.5C4.6 0.5 3.5 1.6 3.5 3V12.1C2.6 12.8 2 13.9 2 15C2 17.2 3.8 19 6 19C8.2 19 10 17.2 10 15C10 13.9 9.4 12.8 8.5 12.1V3C8.5 1.6 7.4 0.5 6 0.5Z" stroke={fillColor} strokeWidth="1" fill="none"/>
  622. </svg>
  623. );
  624. }
  625. // Empty thermometer - heater is OFF
  626. return (
  627. <svg className={className} viewBox="0 0 12 20" fill="none" xmlns="http://www.w3.org/2000/svg">
  628. <path d="M6 0.5C4.6 0.5 3.5 1.6 3.5 3V12.1C2.6 12.8 2 13.9 2 15C2 17.2 3.8 19 6 19C8.2 19 10 17.2 10 15C10 13.9 9.4 12.8 8.5 12.1V3C8.5 1.6 7.4 0.5 6 0.5Z" stroke={fillColor} strokeWidth="1" fill="none"/>
  629. <circle cx="6" cy="15" r="2.5" stroke={fillColor} strokeWidth="1" fill="none"/>
  630. </svg>
  631. );
  632. }
  633. // Humidity indicator with water drop that fills based on level (Bambu Lab style)
  634. // Reference: https://github.com/theicedmango/bambu-humidity
  635. interface HumidityIndicatorProps {
  636. humidity: number | string;
  637. goodThreshold?: number; // <= this is green
  638. fairThreshold?: number; // <= this is orange, > is red
  639. onClick?: () => void;
  640. compact?: boolean; // Smaller version for grid layout
  641. }
  642. function HumidityIndicator({ humidity, goodThreshold = 40, fairThreshold = 60, onClick, compact }: HumidityIndicatorProps) {
  643. const humidityValue = typeof humidity === 'string' ? parseInt(humidity, 10) : humidity;
  644. const good = typeof goodThreshold === 'number' ? goodThreshold : 40;
  645. const fair = typeof fairThreshold === 'number' ? fairThreshold : 60;
  646. // Status thresholds (configurable via settings)
  647. // Good: ≤goodThreshold (green #22a352), Fair: ≤fairThreshold (gold #d4a017), Bad: >fairThreshold (red #c62828)
  648. let textColor: string;
  649. let statusText: string;
  650. if (isNaN(humidityValue)) {
  651. textColor = '#C3C2C1';
  652. statusText = 'Unknown';
  653. } else if (humidityValue <= good) {
  654. textColor = '#22a352'; // Green - Good
  655. statusText = 'Good';
  656. } else if (humidityValue <= fair) {
  657. textColor = '#d4a017'; // Gold - Fair
  658. statusText = 'Fair';
  659. } else {
  660. textColor = '#c62828'; // Red - Bad
  661. statusText = 'Bad';
  662. }
  663. // Fill level based on status: Good=Empty (dry), Fair=Half, Bad=Full (wet)
  664. let DropComponent: React.FC<{ className?: string }>;
  665. if (isNaN(humidityValue)) {
  666. DropComponent = WaterDropEmpty;
  667. } else if (humidityValue <= good) {
  668. DropComponent = WaterDropEmpty; // Good - empty drop (dry)
  669. } else if (humidityValue <= fair) {
  670. DropComponent = WaterDropHalf; // Fair - half filled
  671. } else {
  672. DropComponent = WaterDropFull; // Bad - full (too humid)
  673. }
  674. return (
  675. <button
  676. type="button"
  677. onClick={onClick}
  678. className={`flex items-center gap-1 ${onClick ? 'cursor-pointer hover:opacity-80 transition-opacity' : ''}`}
  679. title={`Humidity: ${humidityValue}% - ${statusText}${onClick ? ' (click for history)' : ''}`}
  680. >
  681. <DropComponent className={compact ? "w-2.5 h-3" : "w-3 h-4"} />
  682. <span className={`font-medium tabular-nums ${compact ? 'text-[10px]' : 'text-xs'}`} style={{ color: textColor }}>{humidityValue}%</span>
  683. </button>
  684. );
  685. }
  686. // Temperature indicator with dynamic icon and coloring
  687. interface TemperatureIndicatorProps {
  688. temp: number;
  689. goodThreshold?: number; // <= this is blue
  690. fairThreshold?: number; // <= this is orange, > is red
  691. onClick?: () => void;
  692. compact?: boolean; // Smaller version for grid layout
  693. }
  694. function TemperatureIndicator({ temp, goodThreshold = 28, fairThreshold = 35, onClick, compact }: TemperatureIndicatorProps) {
  695. // Ensure thresholds are numbers
  696. const good = typeof goodThreshold === 'number' ? goodThreshold : 28;
  697. const fair = typeof fairThreshold === 'number' ? fairThreshold : 35;
  698. let textColor: string;
  699. let statusText: string;
  700. let ThermoComponent: React.FC<{ className?: string }>;
  701. if (temp <= good) {
  702. textColor = '#22a352'; // Green - good (same as humidity)
  703. statusText = 'Good';
  704. ThermoComponent = ThermometerEmpty;
  705. } else if (temp <= fair) {
  706. textColor = '#d4a017'; // Gold - fair (same as humidity)
  707. statusText = 'Fair';
  708. ThermoComponent = ThermometerHalf;
  709. } else {
  710. textColor = '#c62828'; // Red - bad (same as humidity)
  711. statusText = 'Bad';
  712. ThermoComponent = ThermometerFull;
  713. }
  714. return (
  715. <button
  716. type="button"
  717. onClick={onClick}
  718. className={`flex items-center gap-1 ${onClick ? 'cursor-pointer hover:opacity-80 transition-opacity' : ''}`}
  719. title={`Temperature: ${temp}°C - ${statusText}${onClick ? ' (click for history)' : ''}`}
  720. >
  721. <ThermoComponent className={compact ? "w-2.5 h-3" : "w-3 h-4"} />
  722. <span className={`tabular-nums text-right ${compact ? 'text-[10px] w-8' : 'w-12'}`} style={{ color: textColor }}>{temp}°C</span>
  723. </button>
  724. );
  725. }
  726. function getAmsLabel(amsId: number | string, trayCount: number): string {
  727. // Ensure amsId is a number (backend might send string)
  728. const id = typeof amsId === 'string' ? parseInt(amsId, 10) : amsId;
  729. const safeId = isNaN(id) ? 0 : id;
  730. const isHt = trayCount === 1;
  731. // AMS-HT uses IDs starting at 128, regular AMS uses 0-3
  732. const normalizedId = safeId >= 128 ? safeId - 128 : safeId;
  733. const letter = String.fromCharCode(65 + normalizedId); // 0=A, 1=B, 2=C, 3=D
  734. return isHt ? `HT-${letter}` : `AMS-${letter}`;
  735. }
  736. /** Classify an empty AMS slot for UI rendering (#1322 follow-up).
  737. *
  738. * "physical" — firmware positively confirmed no spool (state 9 or 10). The
  739. * bambu_mqtt handler now promotes tray_exist_bits=0 slots to state=9, so
  740. * every empty-by-bitmask slot lands here regardless of firmware payload
  741. * shape.
  742. *
  743. * "reset" — tray_type is missing/empty but firmware hasn't confirmed
  744. * emptiness (state is null, 3, or any non-9/10 value). Typically a slot
  745. * the user cleared with "Reset Slot" where a physical spool may still be
  746. * loaded but unassigned.
  747. *
  748. * Returns null when the slot is loaded (tray_type is present).
  749. */
  750. function getEmptySlotKind(tray: { tray_type?: string | null; state?: number | null } | null | undefined): 'physical' | 'reset' | null {
  751. if (tray?.tray_type) return null;
  752. return (tray?.state === 9 || tray?.state === 10) ? 'physical' : 'reset';
  753. }
  754. function CoverImage({ url, printName }: { url: string | null; printName?: string }) {
  755. const { t } = useTranslation();
  756. const [loaded, setLoaded] = useState(false);
  757. const [error, setError] = useState(false);
  758. const [showOverlay, setShowOverlay] = useState(false);
  759. // Cache-bust the image URL when the print name changes so the browser
  760. // fetches the new cover instead of serving the stale cached image.
  761. const cacheBustedUrl = useMemo(() => {
  762. if (!url) return null;
  763. const sep = url.includes('?') ? '&' : '?';
  764. return withStreamToken(`${url}${sep}v=${encodeURIComponent(printName || Date.now().toString())}`);
  765. }, [url, printName]);
  766. // Reset loaded/error state when the image URL changes
  767. useEffect(() => {
  768. setLoaded(false);
  769. setError(false);
  770. }, [cacheBustedUrl]);
  771. return (
  772. <>
  773. <div
  774. className={`w-20 h-20 flex-shrink-0 rounded-lg overflow-hidden bg-bambu-dark-tertiary flex items-center justify-center ${cacheBustedUrl && loaded ? 'cursor-pointer' : ''}`}
  775. onClick={() => cacheBustedUrl && loaded && setShowOverlay(true)}
  776. >
  777. {cacheBustedUrl && !error ? (
  778. <>
  779. <img
  780. src={cacheBustedUrl}
  781. alt={t('printers.printPreview')}
  782. className={`w-full h-full object-cover ${loaded ? 'block' : 'hidden'}`}
  783. onLoad={() => setLoaded(true)}
  784. onError={() => setError(true)}
  785. />
  786. {!loaded && <Box className="w-8 h-8 text-bambu-gray" />}
  787. </>
  788. ) : (
  789. <Box className="w-8 h-8 text-bambu-gray" />
  790. )}
  791. </div>
  792. {/* Cover Image Overlay */}
  793. {showOverlay && cacheBustedUrl && (
  794. <div
  795. className="fixed inset-0 bg-black/80 flex items-center justify-center z-50 p-8"
  796. onClick={() => setShowOverlay(false)}
  797. >
  798. <div className="relative max-w-2xl max-h-full">
  799. <img
  800. src={cacheBustedUrl}
  801. alt={t('printers.printPreview')}
  802. className="max-w-full max-h-[80vh] rounded-lg shadow-2xl"
  803. />
  804. {printName && (
  805. <p className="text-white text-center mt-4 text-lg">{printName}</p>
  806. )}
  807. </div>
  808. </div>
  809. )}
  810. </>
  811. );
  812. }
  813. interface PrinterMaintenanceInfo {
  814. due_count: number;
  815. warning_count: number;
  816. total_print_hours: number;
  817. }
  818. // Status summary bar component - uses queryClient to read cached statuses
  819. function StatusSummaryBar({ printers }: { printers: Printer[] | undefined }) {
  820. const { t } = useTranslation();
  821. const queryClient = useQueryClient();
  822. // Subscribe to query cache changes to re-render when status updates
  823. // Throttled to prevent rapid re-renders from causing tab crashes
  824. const [cacheTick, setCacheTick] = useState(0);
  825. useEffect(() => {
  826. let pending = false;
  827. const unsubscribe = queryClient.getQueryCache().subscribe(() => {
  828. if (!pending) {
  829. pending = true;
  830. requestAnimationFrame(() => {
  831. setCacheTick(t => t + 1);
  832. pending = false;
  833. });
  834. }
  835. });
  836. return () => unsubscribe();
  837. }, [queryClient]);
  838. const { counts, nextFinish } = useMemo(() => {
  839. let printing = 0;
  840. let paused = 0;
  841. let finished = 0;
  842. let idle = 0;
  843. let offline = 0;
  844. let loading = 0;
  845. let error = 0;
  846. let nextPrinterName: string | null = null;
  847. let nextRemainingMin: number | null = null;
  848. let nextProgress: number = 0;
  849. printers?.forEach((printer) => {
  850. const status = queryClient.getQueryData<{ connected: boolean; state: string | null; remaining_time: number | null; progress: number | null; hms_errors?: HMSError[] }>(['printerStatus', printer.id]);
  851. if (status === undefined) {
  852. // Status not yet loaded - don't count as offline yet
  853. loading++;
  854. } else if (!status.connected) {
  855. offline++;
  856. } else {
  857. // Count printers with active HMS errors as problems
  858. const knownHmsCount =
  859. status.hms_errors ? filterKnownHMSErrors(status.hms_errors).length : 0;
  860. if (knownHmsCount > 0) {
  861. error++;
  862. }
  863. switch (status.state) {
  864. case 'RUNNING':
  865. printing++;
  866. if (status.remaining_time != null && status.remaining_time > 0) {
  867. if (nextRemainingMin === null || status.remaining_time < nextRemainingMin) {
  868. nextRemainingMin = status.remaining_time;
  869. nextPrinterName = printer.name;
  870. nextProgress = status.progress || 0;
  871. }
  872. }
  873. break;
  874. case 'PAUSE':
  875. paused++;
  876. break;
  877. case 'FINISH':
  878. finished++;
  879. break;
  880. case 'FAILED':
  881. // FAILED is the printer's terminal gcode_state after a print stops —
  882. // including user cancellations, where there's no actual fault. Only
  883. // count it as a "problem" when an HMS error is also active; otherwise
  884. // it's just a print that ended unsuccessfully and the plate needs
  885. // clearing (same as FINISH from the operator's perspective).
  886. if (knownHmsCount > 0) {
  887. // Already counted above
  888. } else {
  889. finished++;
  890. }
  891. break;
  892. default:
  893. idle++;
  894. break;
  895. }
  896. }
  897. });
  898. return {
  899. counts: { printing, paused, finished, idle, offline, loading, error, total: (printers?.length || 0) },
  900. nextFinish: nextPrinterName && nextRemainingMin ? { name: nextPrinterName, remainingMin: nextRemainingMin, progress: nextProgress } : null,
  901. };
  902. // eslint-disable-next-line react-hooks/exhaustive-deps
  903. }, [printers, queryClient, cacheTick]);
  904. if (!printers?.length) return null;
  905. const badges: { count: number; dot: string; label: string }[] = [
  906. { count: counts.printing, dot: 'bg-bambu-green animate-pulse', label: t('printers.status.printing').toLowerCase() },
  907. { count: counts.paused, dot: 'bg-status-warning', label: t('printers.status.paused', 'paused').toLowerCase() },
  908. { count: counts.finished, dot: 'bg-blue-400', label: t('printers.status.finished', 'finished').toLowerCase() },
  909. { count: counts.idle, dot: counts.idle > 0 ? 'bg-bambu-green' : 'bg-gray-500', label: t('printers.status.available').toLowerCase() },
  910. { count: counts.error, dot: 'bg-status-error', label: t('printers.status.problem').toLowerCase() },
  911. { count: counts.offline, dot: 'bg-gray-400', label: t('printers.status.offline').toLowerCase() },
  912. ];
  913. return (
  914. <div className="mt-1 flex flex-wrap items-center gap-4 gap-y-2 text-bambu-gray">
  915. {badges.map(({ count, dot, label }) => count > 0 && (
  916. <div key={label} className="flex items-center gap-1.5">
  917. <div className={`w-2 h-2 rounded-full ${dot}`} />
  918. <span className="text-bambu-gray">
  919. <span className="text-white font-medium">{count}</span> {label}
  920. </span>
  921. </div>
  922. ))}
  923. {nextFinish && (
  924. <>
  925. <div className="w-px h-4 bg-bambu-dark-tertiary" />
  926. <div className="flex flex-col gap-1 sm:flex-row sm:items-center sm:gap-2">
  927. <div className="flex items-center gap-2">
  928. <span className="text-bambu-green font-medium">{t('printers.nextAvailable')}:</span>
  929. <span className="text-white font-medium">{nextFinish.name}</span>
  930. </div>
  931. <div className="flex items-center gap-2 w-full sm:w-auto">
  932. <div className="w-full sm:w-16 bg-bambu-dark-tertiary rounded-full h-1.5">
  933. <div
  934. className="bg-bambu-green h-1.5 rounded-full transition-all"
  935. style={{ width: `${nextFinish.progress}%` }}
  936. />
  937. </div>
  938. <span className="text-white font-medium">{Math.round(nextFinish.progress)}%</span>
  939. <span className="text-bambu-gray">({formatDuration(nextFinish.remainingMin * 60)})</span>
  940. </div>
  941. </div>
  942. </>
  943. )}
  944. </div>
  945. );
  946. }
  947. type SortOption = 'name' | 'status' | 'model' | 'location';
  948. type ViewMode = 'expanded' | 'compact';
  949. type ToolbarDropdownOption<T extends string> = {
  950. value: T;
  951. label: string;
  952. };
  953. function ToolbarDropdown<T extends string>({
  954. value,
  955. options,
  956. onChange,
  957. fullWidth = false,
  958. }: {
  959. value: T;
  960. options: ToolbarDropdownOption<T>[];
  961. onChange: (value: T) => void;
  962. fullWidth?: boolean;
  963. }) {
  964. const [isOpen, setIsOpen] = useState(false);
  965. const selectedOption = options.find(option => option.value === value) ?? options[0];
  966. return (
  967. <div className={`relative ${fullWidth ? 'w-full min-w-0' : ''}`}>
  968. <button
  969. type="button"
  970. onClick={() => setIsOpen(open => !open)}
  971. className={`h-8 px-2 rounded-lg border bg-bambu-dark border-bambu-dark-tertiary text-white text-sm font-medium transition-colors hover:bg-bambu-dark-tertiary focus:outline-none focus:border-bambu-green flex items-center justify-between gap-2 ${fullWidth ? 'w-full' : 'min-w-28'}`}
  972. >
  973. <span className="truncate">{selectedOption?.label}</span>
  974. <ChevronDown className={`w-4 h-4 text-bambu-gray transition-transform ${isOpen ? 'rotate-180' : ''}`} />
  975. </button>
  976. {isOpen && (
  977. <>
  978. <div className="fixed inset-0 z-10" onClick={() => setIsOpen(false)} />
  979. <div className="absolute left-0 top-full z-20 mt-1 min-w-full rounded-lg border border-bambu-dark-tertiary bg-bambu-dark-secondary py-1 shadow-xl">
  980. {options.map(option => (
  981. <button
  982. key={option.value}
  983. type="button"
  984. onClick={() => {
  985. onChange(option.value);
  986. setIsOpen(false);
  987. }}
  988. className={`w-full px-3 py-2 text-left text-sm transition-colors hover:bg-bambu-dark-tertiary ${
  989. option.value === value ? 'text-bambu-green' : 'text-white'
  990. }`}
  991. >
  992. {option.label}
  993. </button>
  994. ))}
  995. </div>
  996. </>
  997. )}
  998. </div>
  999. );
  1000. }
  1001. function ToolbarMenu({
  1002. label,
  1003. icon,
  1004. children,
  1005. }: {
  1006. label: string;
  1007. icon: React.ReactNode;
  1008. children: React.ReactNode;
  1009. }) {
  1010. const [isOpen, setIsOpen] = useState(false);
  1011. return (
  1012. <div className="relative">
  1013. <button
  1014. type="button"
  1015. onClick={() => setIsOpen(open => !open)}
  1016. className="h-8 w-8 rounded-lg border bg-bambu-dark border-bambu-dark-tertiary text-white hover:bg-bambu-dark-tertiary transition-colors flex items-center justify-center"
  1017. aria-label={label}
  1018. title={label}
  1019. >
  1020. {icon}
  1021. </button>
  1022. {isOpen && (
  1023. <>
  1024. <div className="fixed inset-0 z-10" onClick={() => setIsOpen(false)} />
  1025. <div className="absolute right-0 top-full z-20 mt-1 min-w-40 rounded-lg border border-bambu-dark-tertiary bg-bambu-dark-secondary p-2 shadow-xl">
  1026. {children}
  1027. </div>
  1028. </>
  1029. )}
  1030. </div>
  1031. );
  1032. }
  1033. const STATUS_GROUP_ORDER: string[] = ['error', 'printing', 'paused', 'finished', 'idle', 'offline'];
  1034. const STATUS_GROUP_META: Record<string, { labelKey: string; dot: string }> = {
  1035. error: { labelKey: 'printers.status.problem', dot: 'bg-status-error' },
  1036. printing: { labelKey: 'printers.status.printing', dot: 'bg-bambu-green animate-pulse' },
  1037. paused: { labelKey: 'printers.status.paused', dot: 'bg-status-warning' },
  1038. finished: { labelKey: 'printers.status.finished', dot: 'bg-blue-400' },
  1039. idle: { labelKey: 'printers.status.idle', dot: 'bg-bambu-green' },
  1040. offline: { labelKey: 'printers.status.offline', dot: 'bg-gray-400' },
  1041. };
  1042. /** Classify a printer into one of the UI status buckets. */
  1043. function classifyPrinterStatus(
  1044. status: { connected: boolean; state: string | null; hms_errors?: HMSError[] } | undefined,
  1045. ): PrinterState {
  1046. if (!status?.connected) return 'offline';
  1047. const hmsErrors = status.hms_errors ? filterKnownHMSErrors(status.hms_errors) : [];
  1048. if (hmsErrors.length > 0) return 'error';
  1049. switch (status.state) {
  1050. case 'RUNNING': return 'printing';
  1051. case 'PAUSE': return 'paused';
  1052. case 'FINISH': return 'finished';
  1053. // FAILED without an active HMS error is the printer's terminal state after
  1054. // any unsuccessful end — including user-cancellations. Treat the same as
  1055. // FINISH for grouping/badging purposes; only escalate to "error" when an
  1056. // HMS code is actually attached (handled by the early-return above).
  1057. case 'FAILED': return 'finished';
  1058. default: return 'idle';
  1059. }
  1060. }
  1061. /**
  1062. * Get human-readable status display text for a printer.
  1063. * Uses stg_cur_name for detailed calibration/preparation stages,
  1064. * otherwise formats the gcode_state nicely.
  1065. */
  1066. function getStatusDisplay(state: string | null | undefined, stg_cur_name: string | null | undefined): string {
  1067. // If we have a specific stage name (calibration, heating, etc.), use it
  1068. if (stg_cur_name) {
  1069. return stg_cur_name;
  1070. }
  1071. // Format the gcode_state nicely
  1072. switch (state) {
  1073. case 'RUNNING':
  1074. return 'Printing';
  1075. case 'PAUSE':
  1076. return 'Paused';
  1077. case 'FINISH':
  1078. return 'Finished';
  1079. case 'FAILED':
  1080. return 'Failed';
  1081. case 'IDLE':
  1082. return 'Idle';
  1083. default:
  1084. return state ? state.charAt(0) + state.slice(1).toLowerCase() : 'Idle';
  1085. }
  1086. }
  1087. // Map SSDP model codes to display names
  1088. function mapModelCode(ssdpModel: string | null): string {
  1089. if (!ssdpModel) return '';
  1090. const modelMap: Record<string, string> = {
  1091. // H2 Series
  1092. 'O1D': 'H2D',
  1093. 'O1E': 'H2D Pro',
  1094. 'O2D': 'H2D Pro',
  1095. 'O1C': 'H2C',
  1096. 'O1C2': 'H2C',
  1097. 'O1S': 'H2S',
  1098. // X1 Series
  1099. 'BL-P001': 'X1C',
  1100. 'BL-P002': 'X1',
  1101. 'BL-P003': 'X1E',
  1102. // X2 Series
  1103. 'N6': 'X2D',
  1104. // P Series
  1105. 'C11': 'P1S',
  1106. 'C12': 'P1P',
  1107. 'C13': 'P2S',
  1108. // A1 Series
  1109. 'N2S': 'A1',
  1110. 'N1': 'A1 Mini',
  1111. // Direct matches
  1112. 'X1C': 'X1C',
  1113. 'X1': 'X1',
  1114. 'X1E': 'X1E',
  1115. 'X2D': 'X2D',
  1116. 'P1S': 'P1S',
  1117. 'P1P': 'P1P',
  1118. 'P2S': 'P2S',
  1119. 'A1': 'A1',
  1120. 'A1 Mini': 'A1 Mini',
  1121. 'H2D': 'H2D',
  1122. 'H2D Pro': 'H2D Pro',
  1123. 'H2C': 'H2C',
  1124. 'H2S': 'H2S',
  1125. };
  1126. return modelMap[ssdpModel] || ssdpModel;
  1127. }
  1128. // ─── AMS Name Hover Card ──────────────────────────────────────────────────────
  1129. // Wraps the AMS label (e.g. "AMS-A") and shows a popup with:
  1130. // • User-defined friendly name (editable, protected by printers:update)
  1131. // • AMS serial number
  1132. // • AMS firmware version
  1133. export function AmsNameHoverCard({
  1134. ams,
  1135. printerId,
  1136. label,
  1137. amsLabels,
  1138. canEdit,
  1139. onSaved,
  1140. children,
  1141. }: {
  1142. ams: import('../api/client').AMSUnit;
  1143. printerId: number;
  1144. label: string; // auto-generated label, e.g. "AMS-A"
  1145. amsLabels?: Record<number, string>;
  1146. canEdit: boolean;
  1147. onSaved: () => void;
  1148. children: React.ReactNode;
  1149. }) {
  1150. const { t } = useTranslation();
  1151. const [isVisible, setIsVisible] = useState(false);
  1152. const [position, setPosition] = useState<'top' | 'bottom'>('top');
  1153. const [editValue, setEditValue] = useState('');
  1154. const [isSaving, setIsSaving] = useState(false);
  1155. const [saveError, setSaveError] = useState<string | null>(null);
  1156. const [isInputFocused, setIsInputFocused] = useState(false);
  1157. const triggerRef = useRef<HTMLDivElement>(null);
  1158. const cardRef = useRef<HTMLDivElement>(null);
  1159. const timeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null);
  1160. useEffect(() => {
  1161. if (isVisible) {
  1162. setEditValue(amsLabels?.[ams.id] ?? '');
  1163. setSaveError(null);
  1164. requestAnimationFrame(() => {
  1165. if (triggerRef.current && cardRef.current) {
  1166. const rect = triggerRef.current.getBoundingClientRect();
  1167. const spaceAbove = rect.top - 56;
  1168. const spaceBelow = window.innerHeight - rect.bottom;
  1169. setPosition(spaceAbove < cardRef.current.offsetHeight + 12 && spaceBelow > spaceAbove ? 'bottom' : 'top');
  1170. }
  1171. });
  1172. }
  1173. }, [isVisible, amsLabels, ams.id]);
  1174. const handleMouseEnter = () => {
  1175. if (timeoutRef.current) clearTimeout(timeoutRef.current);
  1176. timeoutRef.current = setTimeout(() => setIsVisible(true), 80);
  1177. };
  1178. const handleMouseLeave = () => {
  1179. if (timeoutRef.current) clearTimeout(timeoutRef.current);
  1180. if (!isInputFocused) {
  1181. timeoutRef.current = setTimeout(() => setIsVisible(false), 200);
  1182. }
  1183. };
  1184. useEffect(() => () => { if (timeoutRef.current) clearTimeout(timeoutRef.current); }, []);
  1185. const handleSave = async () => {
  1186. if (!canEdit) return;
  1187. setIsSaving(true);
  1188. setSaveError(null);
  1189. try {
  1190. const trimmed = editValue.trim();
  1191. if (trimmed) {
  1192. await api.saveAmsLabel(printerId, ams.id, trimmed, ams.serial_number);
  1193. } else {
  1194. await api.deleteAmsLabel(printerId, ams.id, ams.serial_number);
  1195. }
  1196. onSaved();
  1197. setIsVisible(false);
  1198. } catch (err) {
  1199. setSaveError(err instanceof Error ? err.message : String(err));
  1200. } finally {
  1201. setIsSaving(false);
  1202. }
  1203. };
  1204. const handleClear = async () => {
  1205. if (!canEdit) return;
  1206. setIsSaving(true);
  1207. setSaveError(null);
  1208. try {
  1209. await api.deleteAmsLabel(printerId, ams.id, ams.serial_number);
  1210. onSaved();
  1211. setIsVisible(false);
  1212. } catch (err) {
  1213. setSaveError(err instanceof Error ? err.message : String(err));
  1214. } finally {
  1215. setIsSaving(false);
  1216. }
  1217. };
  1218. return (
  1219. <div
  1220. ref={triggerRef}
  1221. className="relative inline-block"
  1222. onMouseEnter={handleMouseEnter}
  1223. onMouseLeave={handleMouseLeave}
  1224. >
  1225. {children}
  1226. {isVisible && (
  1227. <div
  1228. ref={cardRef}
  1229. className={`
  1230. absolute left-0 z-50
  1231. ${position === 'top' ? 'bottom-full mb-2' : 'top-full mt-2'}
  1232. animate-in fade-in-0 zoom-in-95 duration-150
  1233. `}
  1234. style={{ maxWidth: 'calc(100vw - 24px)' }}
  1235. onMouseEnter={handleMouseEnter}
  1236. onMouseLeave={handleMouseLeave}
  1237. >
  1238. <div className="w-52 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg shadow-xl overflow-hidden backdrop-blur-sm p-2.5 space-y-2">
  1239. {/* AMS auto-label */}
  1240. <div className="text-[10px] uppercase tracking-wider text-bambu-gray font-medium">{label}</div>
  1241. {/* Serial number */}
  1242. <div className="flex items-center justify-between gap-2">
  1243. <span className="text-[10px] tracking-wide text-bambu-gray font-medium shrink-0">
  1244. {t('printers.amsPopup.serialNumber')}
  1245. </span>
  1246. <span className="text-[10px] text-white font-mono truncate">{ams.serial_number || '—'}</span>
  1247. </div>
  1248. {/* Firmware version */}
  1249. <div className="flex items-center justify-between gap-2">
  1250. <span className="text-[10px] tracking-wide text-bambu-gray font-medium shrink-0">
  1251. {t('printers.amsPopup.firmwareVersion')}
  1252. </span>
  1253. <span className="text-[10px] text-white font-mono truncate">{ams.sw_ver || '—'}</span>
  1254. </div>
  1255. {/* Divider */}
  1256. <div className="h-px bg-bambu-dark-tertiary/50" />
  1257. {/* Friendly name editor */}
  1258. <div className="space-y-1">
  1259. <span className="text-[10px] text-bambu-gray font-medium block">
  1260. {t('printers.amsPopup.friendlyName')}
  1261. </span>
  1262. <input
  1263. type="text"
  1264. value={editValue}
  1265. onChange={(e) => canEdit && setEditValue(e.target.value)}
  1266. onKeyDown={(e) => e.key === 'Enter' && handleSave()}
  1267. onFocus={() => setIsInputFocused(true)}
  1268. onBlur={() => {
  1269. setIsInputFocused(false);
  1270. if (timeoutRef.current) clearTimeout(timeoutRef.current);
  1271. timeoutRef.current = setTimeout(() => setIsVisible(false), 200);
  1272. }}
  1273. placeholder={canEdit ? t('printers.amsPopup.friendlyNamePlaceholder') : (amsLabels?.[ams.id] || '—')}
  1274. disabled={!canEdit}
  1275. title={!canEdit ? t('printers.amsPopup.noEditPermission') : undefined}
  1276. className="w-full bg-bambu-dark border border-bambu-dark-tertiary rounded px-2 py-1 text-xs text-white placeholder-bambu-gray/60 focus:outline-none focus:border-bambu-green disabled:opacity-50 disabled:cursor-not-allowed"
  1277. maxLength={100}
  1278. />
  1279. {canEdit && (
  1280. <div className="space-y-1">
  1281. {saveError && (
  1282. <p className="text-[10px] text-red-400 break-words">{saveError}</p>
  1283. )}
  1284. <div className="flex gap-1 justify-end">
  1285. <button
  1286. onClick={handleSave}
  1287. disabled={isSaving}
  1288. className="px-2 py-0.5 text-[10px] bg-bambu-green text-white rounded hover:bg-bambu-green/80 disabled:opacity-50"
  1289. >
  1290. {t('printers.amsPopup.save')}
  1291. </button>
  1292. {amsLabels?.[ams.id] && (
  1293. <button
  1294. onClick={handleClear}
  1295. disabled={isSaving}
  1296. className="px-2 py-0.5 text-[10px] bg-bambu-dark-tertiary text-bambu-gray rounded hover:bg-bambu-dark-tertiary/70 disabled:opacity-50"
  1297. >
  1298. {t('printers.amsPopup.clear')}
  1299. </button>
  1300. )}
  1301. </div>
  1302. </div>
  1303. )}
  1304. </div>
  1305. </div>
  1306. </div>
  1307. )}
  1308. </div>
  1309. );
  1310. }
  1311. // AMS drying presets from BambuStudio filament profiles (idle mode temps)
  1312. // Format: { n3f temp, n3s temp, n3f hours, n3s hours }
  1313. const DRYING_PRESETS: Record<string, { n3f: number; n3s: number; n3f_hours: number; n3s_hours: number }> = {
  1314. 'PLA': { n3f: 45, n3s: 45, n3f_hours: 12, n3s_hours: 12 },
  1315. 'PETG': { n3f: 65, n3s: 65, n3f_hours: 12, n3s_hours: 12 },
  1316. 'TPU': { n3f: 65, n3s: 75, n3f_hours: 12, n3s_hours: 18 },
  1317. 'ABS': { n3f: 65, n3s: 80, n3f_hours: 12, n3s_hours: 8 },
  1318. 'ASA': { n3f: 65, n3s: 80, n3f_hours: 12, n3s_hours: 8 },
  1319. 'PA': { n3f: 65, n3s: 85, n3f_hours: 12, n3s_hours: 12 },
  1320. 'PC': { n3f: 65, n3s: 80, n3f_hours: 12, n3s_hours: 8 },
  1321. 'PVA': { n3f: 65, n3s: 85, n3f_hours: 12, n3s_hours: 18 },
  1322. };
  1323. function PrinterCard({
  1324. printer,
  1325. hideIfDisconnected,
  1326. maintenanceInfo,
  1327. viewMode = 'expanded',
  1328. cardSize = 2,
  1329. amsThresholds,
  1330. spoolmanEnabled = false,
  1331. linkedSpools,
  1332. spoolmanUrl,
  1333. spoolmanSyncMode,
  1334. onGetAssignment,
  1335. onUnassignSpool,
  1336. spoolmanSpools,
  1337. spoolmanSlotAssignments,
  1338. spoolmanLoading = false,
  1339. onUnassignSpoolmanSpool,
  1340. timeFormat = 'system',
  1341. cameraViewMode = 'window',
  1342. onOpenEmbeddedCamera,
  1343. checkPrinterFirmware = true,
  1344. dryingPresets = DRYING_PRESETS,
  1345. requirePlateClear = false,
  1346. selectionMode = false,
  1347. isSelected = false,
  1348. onToggleSelect,
  1349. }: {
  1350. printer: Printer;
  1351. hideIfDisconnected?: boolean;
  1352. maintenanceInfo?: PrinterMaintenanceInfo;
  1353. viewMode?: ViewMode;
  1354. cardSize?: number;
  1355. amsThresholds?: {
  1356. humidityGood: number;
  1357. humidityFair: number;
  1358. tempGood: number;
  1359. tempFair: number;
  1360. };
  1361. spoolmanEnabled?: boolean;
  1362. hasUnlinkedSpools?: boolean;
  1363. linkedSpools?: Record<string, LinkedSpoolInfo>;
  1364. spoolmanUrl?: string | null;
  1365. spoolmanSyncMode?: string | null;
  1366. spoolAssignments?: SpoolAssignment[];
  1367. onGetAssignment?: (printerId: number, amsId: number, trayId: number) => SpoolAssignment | undefined;
  1368. onUnassignSpool?: (printerId: number, amsId: number, trayId: number) => void;
  1369. spoolmanSpools?: InventorySpool[];
  1370. spoolmanSlotAssignments?: SpoolmanSlotAssignmentRow[];
  1371. spoolmanLoading?: boolean;
  1372. onUnassignSpoolmanSpool?: (spoolmanSpoolId: number) => void;
  1373. timeFormat?: 'system' | '12h' | '24h';
  1374. cameraViewMode?: 'window' | 'embedded';
  1375. onOpenEmbeddedCamera?: (printerId: number, printerName: string) => void;
  1376. checkPrinterFirmware?: boolean;
  1377. dryingPresets?: Record<string, { n3f: number; n3s: number; n3f_hours: number; n3s_hours: number }>;
  1378. requirePlateClear?: boolean;
  1379. selectionMode?: boolean;
  1380. isSelected?: boolean;
  1381. onToggleSelect?: (id: number) => void;
  1382. }) {
  1383. const { t } = useTranslation();
  1384. const queryClient = useQueryClient();
  1385. const navigate = useNavigate();
  1386. const { showToast } = useToast();
  1387. const { hasPermission } = useAuth();
  1388. const [showMenu, setShowMenu] = useState(false);
  1389. const [showDeleteConfirm, setShowDeleteConfirm] = useState(false);
  1390. const [deleteArchives, setDeleteArchives] = useState(true);
  1391. const [showEditModal, setShowEditModal] = useState(false);
  1392. const [showFileManager, setShowFileManager] = useState(false);
  1393. const [showMQTTDebug, setShowMQTTDebug] = useState(false);
  1394. const [showPowerOnConfirm, setShowPowerOnConfirm] = useState(false);
  1395. const [showPowerOffConfirm, setShowPowerOffConfirm] = useState(false);
  1396. const [haToggleConfirm, setHaToggleConfirm] = useState<SmartPlug | null>(null);
  1397. const [showHMSModal, setShowHMSModal] = useState(false);
  1398. const [showStopConfirm, setShowStopConfirm] = useState(false);
  1399. const [showPauseConfirm, setShowPauseConfirm] = useState(false);
  1400. const [showSpeedMenu, setShowSpeedMenu] = useState<number | null>(null);
  1401. const [showAirductMenu, setShowAirductMenu] = useState<number | null>(null);
  1402. const [showBedJogMenu, setShowBedJogMenu] = useState<number | null>(null);
  1403. const [bedJogStep, setBedJogStep] = useState<number>(10);
  1404. const [showNotHomedModal, setShowNotHomedModal] = useState<null | { distance: number }>(null);
  1405. const [showResumeConfirm, setShowResumeConfirm] = useState(false);
  1406. const [showSkipObjectsModal, setShowSkipObjectsModal] = useState(false);
  1407. const [showUploadForPrint, setShowUploadForPrint] = useState(false);
  1408. const [showPrinterInfo, setShowPrinterInfo] = useState(false);
  1409. const [showDiagnostic, setShowDiagnostic] = useState(false);
  1410. const closePrinterInfo = useCallback(() => setShowPrinterInfo(false), []);
  1411. const [printAfterUpload, setPrintAfterUpload] = useState<{ id: number; filename: string } | null>(null);
  1412. // AMS drying popover state: which AMS unit has the popover open
  1413. const [dryingPopoverAmsId, setDryingPopoverAmsId] = useState<number | null>(null);
  1414. const [dryingPopoverModuleType, setDryingPopoverModuleType] = useState<string>('n3f');
  1415. const [dryingFilament, setDryingFilament] = useState('PLA');
  1416. const [dryingTemp, setDryingTemp] = useState(50);
  1417. const [dryingDuration, setDryingDuration] = useState(4);
  1418. const [dryingRotateTray, setDryingRotateTray] = useState(false);
  1419. const [dryingPopoverPos, setDryingPopoverPos] = useState<{ top: number; left: number } | null>(null);
  1420. const [isDraggingFile, setIsDraggingFile] = useState(false);
  1421. const [isDropUploading, setIsDropUploading] = useState(false);
  1422. const dragCounterRef = useRef(0);
  1423. const [amsHistoryModal, setAmsHistoryModal] = useState<{
  1424. amsId: number;
  1425. amsLabel: string;
  1426. mode: 'humidity' | 'temperature';
  1427. } | null>(null);
  1428. const [linkSpoolModal, setLinkSpoolModal] = useState<{
  1429. tagUid: string;
  1430. trayUuid: string;
  1431. printerId: number;
  1432. amsId: number;
  1433. trayId: number;
  1434. } | null>(null);
  1435. const [assignSpoolModal, setAssignSpoolModal] = useState<{
  1436. printerId: number;
  1437. amsId: number;
  1438. trayId: number;
  1439. trayInfo: { type: string; color: string; location: string; material?: string; profile?: string };
  1440. } | null>(null);
  1441. const [configureSlotModal, setConfigureSlotModal] = useState<{
  1442. amsId: number;
  1443. trayId: number;
  1444. trayCount: number;
  1445. trayType?: string;
  1446. trayColor?: string;
  1447. traySubBrands?: string;
  1448. trayInfoIdx?: string;
  1449. extruderId?: number;
  1450. caliIdx?: number | null;
  1451. savedPresetId?: string;
  1452. } | null>(null);
  1453. const [showFirmwareModal, setShowFirmwareModal] = useState(false);
  1454. const [plateCheckResult, setPlateCheckResult] = useState<{
  1455. is_empty: boolean;
  1456. confidence: number;
  1457. difference_percent: number;
  1458. message: string;
  1459. debug_image_url?: string;
  1460. needs_calibration: boolean;
  1461. light_warning?: boolean;
  1462. reference_count?: number;
  1463. max_references?: number;
  1464. roi?: { x: number; y: number; w: number; h: number };
  1465. } | null>(null);
  1466. const [isCheckingPlate, setIsCheckingPlate] = useState(false);
  1467. const [isCalibrating, setIsCalibrating] = useState(false);
  1468. const [editingRoi, setEditingRoi] = useState<{ x: number; y: number; w: number; h: number } | null>(null);
  1469. const [isSavingRoi, setIsSavingRoi] = useState(false);
  1470. const [plateCheckLightWasOff, setPlateCheckLightWasOff] = useState(false);
  1471. const { data: status } = useQuery({
  1472. queryKey: ['printerStatus', printer.id],
  1473. queryFn: () => api.getPrinterStatus(printer.id),
  1474. refetchInterval: 30000, // Fallback polling, WebSocket handles real-time
  1475. });
  1476. // Check for firmware updates (cached for 5 minutes, can be disabled in settings)
  1477. const { data: firmwareInfo } = useQuery({
  1478. queryKey: ['firmwareUpdate', printer.id],
  1479. queryFn: () => firmwareApi.checkPrinterUpdate(printer.id),
  1480. staleTime: 5 * 60 * 1000,
  1481. refetchInterval: 5 * 60 * 1000,
  1482. enabled: checkPrinterFirmware && hasPermission('firmware:read'),
  1483. });
  1484. // Collect unique tray_info_idx values for cloud filament info lookup
  1485. const trayInfoIds = useMemo(() => {
  1486. const ids = new Set<string>();
  1487. if (status?.ams) {
  1488. for (const ams of status.ams) {
  1489. for (const tray of ams.tray || []) {
  1490. if (tray.tray_info_idx) {
  1491. ids.add(tray.tray_info_idx);
  1492. }
  1493. }
  1494. }
  1495. }
  1496. for (const vt of status?.vt_tray ?? []) {
  1497. if (vt.tray_info_idx) ids.add(vt.tray_info_idx);
  1498. }
  1499. if (status?.nozzle_rack) {
  1500. for (const slot of status.nozzle_rack) {
  1501. if (slot.filament_id) {
  1502. ids.add(slot.filament_id);
  1503. }
  1504. }
  1505. }
  1506. return Array.from(ids);
  1507. }, [status?.ams, status?.vt_tray, status?.nozzle_rack]);
  1508. // Collect loaded filament types for queue widget filtering
  1509. const loadedFilamentTypes = useMemo(() => {
  1510. const types = new Set<string>();
  1511. if (status?.ams) {
  1512. for (const ams of status.ams) {
  1513. for (const tray of ams.tray || []) {
  1514. if (tray.tray_type) types.add(tray.tray_type.toUpperCase());
  1515. }
  1516. }
  1517. }
  1518. for (const vt of status?.vt_tray ?? []) {
  1519. if (vt.tray_type) types.add(vt.tray_type.toUpperCase());
  1520. }
  1521. return types;
  1522. }, [status?.ams, status?.vt_tray]);
  1523. // Collect loaded filament type+color pairs for queue widget override matching
  1524. // Format: "TYPE:rrggbb" (e.g., "PETG:ffffff") — mirrors backend _count_override_color_matches()
  1525. const loadedFilaments = useMemo(() => {
  1526. const filaments = new Set<string>();
  1527. if (status?.ams) {
  1528. for (const ams of status.ams) {
  1529. for (const tray of ams.tray || []) {
  1530. if (tray.tray_type && tray.tray_color) {
  1531. const color = tray.tray_color.replace('#', '').toLowerCase().slice(0, 6);
  1532. filaments.add(`${tray.tray_type.toUpperCase()}:${color}`);
  1533. }
  1534. }
  1535. }
  1536. }
  1537. for (const vt of status?.vt_tray ?? []) {
  1538. if (vt.tray_type && vt.tray_color) {
  1539. const color = vt.tray_color.replace('#', '').toLowerCase().slice(0, 6);
  1540. filaments.add(`${vt.tray_type.toUpperCase()}:${color}`);
  1541. }
  1542. }
  1543. return filaments;
  1544. }, [status?.ams, status?.vt_tray]);
  1545. // Fetch cloud filament info for tooltips (name includes color, also has K value)
  1546. const { data: filamentInfo } = useQuery({
  1547. queryKey: ['filamentInfo', trayInfoIds],
  1548. queryFn: () => api.getFilamentInfo(trayInfoIds),
  1549. enabled: trayInfoIds.length > 0,
  1550. staleTime: 5 * 60 * 1000, // 5 minutes
  1551. });
  1552. // Fetch slot preset mappings (stores preset name for user-configured slots)
  1553. const { data: slotPresets } = useQuery({
  1554. queryKey: ['slotPresets', printer.id],
  1555. queryFn: () => api.getSlotPresets(printer.id),
  1556. staleTime: 2 * 60 * 1000, // 2 minutes
  1557. });
  1558. // Fetch plate list for the archive linked to the active print (#881 follow-up).
  1559. // Only queried when there's a running print backed by an archive; shared
  1560. // React Query cache with the Queue / Archives pages keeps it cheap.
  1561. const activeArchiveId =
  1562. (status?.state === 'RUNNING' || status?.state === 'PAUSE') ? status?.current_archive_id ?? null : null;
  1563. const { data: activeArchivePlates } = useQuery({
  1564. queryKey: ['archive-plates', activeArchiveId],
  1565. queryFn: () => api.getArchivePlates(activeArchiveId!),
  1566. enabled: activeArchiveId != null,
  1567. staleTime: 5 * 60 * 1000,
  1568. });
  1569. const activePlateLabel = (() => {
  1570. if (!activeArchivePlates?.is_multi_plate || status?.current_plate_id == null) return null;
  1571. const plate = activeArchivePlates.plates.find(p => p.index === status.current_plate_id);
  1572. return plate?.name || t('printers.plateNumber', 'Plate {{number}}', { number: status.current_plate_id });
  1573. })();
  1574. // Fetch user-defined AMS friendly names from the database
  1575. const { data: amsLabels, refetch: refetchAmsLabels } = useQuery({
  1576. queryKey: ['amsLabels', printer.id],
  1577. queryFn: () => api.getAmsLabels(printer.id),
  1578. staleTime: 5 * 60 * 1000, // 5 minutes
  1579. });
  1580. // Cache WiFi signal to prevent it disappearing on updates
  1581. const [cachedWifiSignal, setCachedWifiSignal] = useState<number | null>(null);
  1582. useEffect(() => {
  1583. if (status?.wifi_signal != null) {
  1584. setCachedWifiSignal(status.wifi_signal);
  1585. }
  1586. }, [status?.wifi_signal]);
  1587. const wifiSignal = status?.wifi_signal ?? cachedWifiSignal;
  1588. // Cache connected state to prevent flicker when status briefly becomes undefined
  1589. const cachedConnected = useRef<boolean | undefined>(undefined);
  1590. useEffect(() => {
  1591. if (status?.connected !== undefined) {
  1592. cachedConnected.current = status.connected;
  1593. }
  1594. }, [status?.connected]);
  1595. const isConnected = status?.connected ?? cachedConnected.current;
  1596. // Cache ams_extruder_map to prevent L/R indicators bouncing on updates
  1597. const cachedAmsExtruderMap = useRef<Record<string, number>>({});
  1598. useEffect(() => {
  1599. if (status?.ams_extruder_map && Object.keys(status.ams_extruder_map).length > 0) {
  1600. cachedAmsExtruderMap.current = status.ams_extruder_map;
  1601. }
  1602. }, [status?.ams_extruder_map]);
  1603. const amsExtruderMap = (status?.ams_extruder_map && Object.keys(status.ams_extruder_map).length > 0)
  1604. ? status.ams_extruder_map
  1605. : cachedAmsExtruderMap.current;
  1606. // Cache AMS data to prevent it disappearing on idle/offline printers
  1607. const cachedAmsData = useRef<AMSUnit[]>([]);
  1608. useEffect(() => {
  1609. if (status?.ams && status.ams.length > 0) {
  1610. cachedAmsData.current = status.ams;
  1611. }
  1612. }, [status?.ams]);
  1613. const amsData = (status?.ams && status.ams.length > 0) ? status.ams : cachedAmsData.current;
  1614. // Cache tray_now to prevent flickering when undefined values come in
  1615. // Valid tray IDs: 0-253 for AMS, 254 for external spool
  1616. // tray_now=255 means "no tray loaded" (Bambu protocol sentinel) — never active
  1617. const cachedTrayNow = useRef<number | undefined>(undefined);
  1618. const currentTrayNow = status?.tray_now;
  1619. // Update cache: 255 means "no tray" so clear cache; valid values get cached
  1620. if (currentTrayNow !== undefined && currentTrayNow !== 255) {
  1621. cachedTrayNow.current = currentTrayNow;
  1622. } else if (currentTrayNow === 255) {
  1623. cachedTrayNow.current = undefined;
  1624. }
  1625. const effectiveTrayNow = (currentTrayNow !== undefined && currentTrayNow !== 255)
  1626. ? currentTrayNow
  1627. : cachedTrayNow.current;
  1628. // Fetch smart plug for this printer
  1629. const { data: smartPlug } = useQuery({
  1630. queryKey: ['smartPlugByPrinter', printer.id],
  1631. queryFn: () => api.getSmartPlugByPrinter(printer.id),
  1632. });
  1633. // Fetch script plugs for this printer (for multi-device control)
  1634. const { data: scriptPlugs } = useQuery({
  1635. queryKey: ['scriptPlugsByPrinter', printer.id],
  1636. queryFn: () => api.getScriptPlugsByPrinter(printer.id),
  1637. });
  1638. // Fetch smart plug status if plug exists (faster refresh for energy monitoring)
  1639. const { data: plugStatus } = useQuery({
  1640. queryKey: ['smartPlugStatus', smartPlug?.id],
  1641. queryFn: () => smartPlug ? api.getSmartPlugStatus(smartPlug.id) : null,
  1642. enabled: !!smartPlug,
  1643. refetchInterval: 10000, // 10 seconds for real-time power display
  1644. });
  1645. // Fetch queue count for this printer
  1646. const { data: queueItems } = useQuery({
  1647. queryKey: ['queue', printer.id, 'pending'],
  1648. queryFn: () => api.getQueue(printer.id, 'pending'),
  1649. });
  1650. // Filter queue items by filament compatibility (same logic as PrinterQueueWidget)
  1651. // so the badge only shows on printers that can actually run the queued jobs.
  1652. // An empty Set means no filaments are loaded — jobs requiring specific types are incompatible.
  1653. const queueCount = useMemo(() => {
  1654. if (!queueItems?.length) return 0;
  1655. return filterCompatibleQueueItems(queueItems, loadedFilamentTypes, loadedFilaments).length;
  1656. }, [queueItems, loadedFilamentTypes, loadedFilaments]);
  1657. // Fetch currently printing queue item to show who started it (Issue #206)
  1658. const { data: printingQueueItems } = useQuery({
  1659. queryKey: ['queue', printer.id, 'printing'],
  1660. queryFn: () => api.getQueue(printer.id, 'printing'),
  1661. enabled: status?.state === 'RUNNING',
  1662. });
  1663. // Fetch reprint user info (for prints started via Reprint, not queue - Issue #206)
  1664. const { data: reprintUser } = useQuery({
  1665. queryKey: ['currentPrintUser', printer.id],
  1666. queryFn: () => api.getCurrentPrintUser(printer.id),
  1667. enabled: status?.state === 'RUNNING',
  1668. });
  1669. // Combine both sources: queue item user takes precedence, then reprint user
  1670. const currentPrintUser = printingQueueItems?.[0]?.created_by_username || reprintUser?.username;
  1671. // Fetch last completed print for this printer
  1672. const { data: lastPrints } = useQuery({
  1673. queryKey: ['archives', printer.id, 'last'],
  1674. queryFn: () => api.getArchives(printer.id, 1, 0),
  1675. enabled: status?.connected && status?.state !== 'RUNNING',
  1676. });
  1677. const lastPrint = lastPrints?.[0];
  1678. const isPrintingOrPaused = status?.state === 'RUNNING' || status?.state === 'PAUSE';
  1679. const needsPlateClear = requirePlateClear && status?.awaiting_plate_clear === true;
  1680. const showClearPlateButton = status?.connected && needsPlateClear && !isPrintingOrPaused;
  1681. const plateStatus = (() => {
  1682. if (!requirePlateClear || !status?.connected) return null;
  1683. if (isPrintingOrPaused) {
  1684. return {
  1685. label: t('printers.plateStatus.inUse'),
  1686. className: 'bg-blue-500/20 text-blue-400',
  1687. };
  1688. }
  1689. if (status.awaiting_plate_clear) {
  1690. return {
  1691. label: t('printers.plateStatus.notCleared'),
  1692. className: 'bg-yellow-500/20 text-yellow-400',
  1693. };
  1694. }
  1695. return {
  1696. label: t('printers.plateStatus.cleared'),
  1697. className: 'bg-status-ok/20 text-status-ok',
  1698. };
  1699. })();
  1700. const plateStatusPill = plateStatus ? (
  1701. <span className={`inline-flex flex-shrink-0 items-center rounded-full px-2 py-0.5 text-[10px] font-medium ${plateStatus.className}`}>
  1702. {plateStatus.label}
  1703. </span>
  1704. ) : null;
  1705. // Determine if this card should be hidden (use cached connected state to prevent flicker)
  1706. const shouldHide = hideIfDisconnected && isConnected === false;
  1707. const deleteMutation = useMutation({
  1708. mutationFn: (options: { deleteArchives: boolean }) =>
  1709. api.deletePrinter(printer.id, options.deleteArchives),
  1710. onSuccess: () => {
  1711. queryClient.invalidateQueries({ queryKey: ['printers'] });
  1712. queryClient.invalidateQueries({ queryKey: ['archives'] });
  1713. queryClient.invalidateQueries({ queryKey: ['maintenanceOverview'] });
  1714. },
  1715. onError: (error: Error) => showToast(error.message || t('printers.toast.failedToDelete'), 'error'),
  1716. });
  1717. const connectMutation = useMutation({
  1718. mutationFn: () => api.connectPrinter(printer.id),
  1719. onSuccess: () => {
  1720. queryClient.invalidateQueries({ queryKey: ['printerStatus', printer.id] });
  1721. },
  1722. });
  1723. const forceRefreshMutation = useMutation({
  1724. mutationFn: () => api.refreshPrinterStatus(printer.id),
  1725. onSuccess: () => {
  1726. queryClient.invalidateQueries({ queryKey: ['printerStatus', printer.id] });
  1727. showToast(t('printers.forceRefreshSuccess'), 'success');
  1728. },
  1729. onError: (error: Error) => showToast(error.message || t('printers.toast.failedToSendCommand'), 'error'),
  1730. });
  1731. const unlinkSpoolMutation = useMutation({
  1732. mutationFn: (spoolId: number) => api.unlinkSpool(spoolId),
  1733. onSuccess: (result) => {
  1734. showToast(t('spoolman.unlinkSuccess') || result?.message, 'success');
  1735. queryClient.invalidateQueries({ queryKey: ['linked-spools'] });
  1736. queryClient.invalidateQueries({ queryKey: ['unlinked-spools'] });
  1737. queryClient.invalidateQueries({ queryKey: ['spoolman-slot-assignments'] });
  1738. },
  1739. onError: (error: Error) => {
  1740. showToast(error.message || t('spoolman.unlinkFailed'), 'error');
  1741. },
  1742. });
  1743. // AMS drying mutations
  1744. const startDryingMutation = useMutation({
  1745. mutationFn: ({ amsId, temp, duration, filament, rotateTray }: { amsId: number; temp: number; duration: number; filament: string; rotateTray: boolean }) =>
  1746. api.startDrying(printer.id, amsId, temp, duration, filament, rotateTray),
  1747. onSuccess: () => {
  1748. setDryingPopoverAmsId(null);
  1749. queryClient.invalidateQueries({ queryKey: ['printerStatus', printer.id] });
  1750. },
  1751. onError: (error: Error) => showToast(error.message || t('printers.toast.failedToSendCommand'), 'error'),
  1752. });
  1753. const stopDryingMutation = useMutation({
  1754. mutationFn: (amsId: number) => api.stopDrying(printer.id, amsId),
  1755. onSuccess: () => {
  1756. queryClient.invalidateQueries({ queryKey: ['printerStatus', printer.id] });
  1757. },
  1758. onError: (error: Error) => showToast(error.message || t('printers.toast.failedToSendCommand'), 'error'),
  1759. });
  1760. // Smart plug control mutations
  1761. const powerControlMutation = useMutation({
  1762. mutationFn: (action: 'on' | 'off') =>
  1763. smartPlug ? api.controlSmartPlug(smartPlug.id, action) : Promise.reject('No plug'),
  1764. onSuccess: () => {
  1765. queryClient.invalidateQueries({ queryKey: ['smartPlugStatus', smartPlug?.id] });
  1766. },
  1767. });
  1768. const toggleAutoOffMutation = useMutation({
  1769. mutationFn: (enabled: boolean) =>
  1770. smartPlug ? api.updateSmartPlug(smartPlug.id, { auto_off: enabled }) : Promise.reject('No plug'),
  1771. onSuccess: () => {
  1772. queryClient.invalidateQueries({ queryKey: ['smartPlugByPrinter', printer.id] });
  1773. // Also invalidate the smart-plugs list to keep Settings page in sync
  1774. queryClient.invalidateQueries({ queryKey: ['smart-plugs'] });
  1775. },
  1776. });
  1777. // Run HA entity mutation — scripts use 'on' (trigger), switches use 'toggle'
  1778. const runScriptMutation = useMutation({
  1779. mutationFn: ({ id, action }: { id: number; action: 'on' | 'toggle' }) => api.controlSmartPlug(id, action),
  1780. onSuccess: () => {
  1781. showToast(t('printers.toast.scriptTriggered'));
  1782. },
  1783. onError: (error: Error) => showToast(error.message || t('printers.toast.failedToRunScript'), 'error'),
  1784. });
  1785. // Print control mutations
  1786. const stopPrintMutation = useMutation({
  1787. mutationFn: () => api.stopPrint(printer.id),
  1788. onSuccess: () => {
  1789. showToast(t('printers.toast.printStopped'));
  1790. queryClient.invalidateQueries({ queryKey: ['printerStatus', printer.id] });
  1791. },
  1792. onError: (error: Error) => showToast(error.message || t('printers.toast.failedToStopPrint'), 'error'),
  1793. });
  1794. const pausePrintMutation = useMutation({
  1795. mutationFn: () => api.pausePrint(printer.id),
  1796. onSuccess: () => {
  1797. showToast(t('printers.toast.printPaused'));
  1798. queryClient.invalidateQueries({ queryKey: ['printerStatus', printer.id] });
  1799. },
  1800. onError: (error: Error) => showToast(error.message || t('printers.toast.failedToPausePrint'), 'error'),
  1801. });
  1802. const resumePrintMutation = useMutation({
  1803. mutationFn: () => api.resumePrint(printer.id),
  1804. onSuccess: () => {
  1805. showToast(t('printers.toast.printResumed'));
  1806. queryClient.invalidateQueries({ queryKey: ['printerStatus', printer.id] });
  1807. },
  1808. onError: (error: Error) => showToast(error.message || t('printers.toast.failedToResumePrint'), 'error'),
  1809. });
  1810. const clearPlateMutation = useMutation({
  1811. mutationFn: () => api.clearPlate(printer.id),
  1812. onSuccess: () => {
  1813. showToast(t('queue.clearPlateSuccess'));
  1814. queryClient.setQueryData(['printerStatus', printer.id], (old: PrinterStatus | undefined) =>
  1815. old ? { ...old, awaiting_plate_clear: false } : old
  1816. );
  1817. queryClient.invalidateQueries({ queryKey: ['printerStatus', printer.id] });
  1818. queryClient.invalidateQueries({ queryKey: ['queue', printer.id] });
  1819. },
  1820. onError: (error: Error) => showToast(error.message || t('printers.toast.failedToSendCommand'), 'error'),
  1821. });
  1822. // Chamber light mutation with optimistic update
  1823. const chamberLightMutation = useMutation({
  1824. mutationFn: (on: boolean) => api.setChamberLight(printer.id, on),
  1825. onMutate: async (on) => {
  1826. // Cancel any outgoing refetches
  1827. await queryClient.cancelQueries({ queryKey: ['printerStatus', printer.id] });
  1828. // Snapshot the previous value
  1829. const previousStatus = queryClient.getQueryData(['printerStatus', printer.id]);
  1830. // Optimistically update
  1831. queryClient.setQueryData(['printerStatus', printer.id], (old: typeof status) => ({
  1832. ...old,
  1833. chamber_light: on,
  1834. }));
  1835. return { previousStatus };
  1836. },
  1837. onSuccess: (_, on) => {
  1838. showToast(`Chamber light ${on ? 'on' : 'off'}`);
  1839. },
  1840. onError: (error: Error, _, context) => {
  1841. // Rollback on error
  1842. if (context?.previousStatus) {
  1843. queryClient.setQueryData(['printerStatus', printer.id], context.previousStatus);
  1844. }
  1845. showToast(error.message || t('printers.toast.failedToControlChamberLight'), 'error');
  1846. },
  1847. });
  1848. // Print speed mutation with optimistic update
  1849. const printSpeedMutation = useMutation({
  1850. mutationFn: (mode: number) => api.setPrintSpeed(printer.id, mode),
  1851. onMutate: async (mode) => {
  1852. await queryClient.cancelQueries({ queryKey: ['printerStatus', printer.id] });
  1853. const previousStatus = queryClient.getQueryData(['printerStatus', printer.id]);
  1854. queryClient.setQueryData(['printerStatus', printer.id], (old: typeof status) => ({
  1855. ...old,
  1856. speed_level: mode,
  1857. }));
  1858. return { previousStatus };
  1859. },
  1860. onError: (error: Error, _, context) => {
  1861. if (context?.previousStatus) {
  1862. queryClient.setQueryData(['printerStatus', printer.id], context.previousStatus);
  1863. }
  1864. showToast(error.message || t('printers.toast.failedToSetSpeed'), 'error');
  1865. },
  1866. });
  1867. const airductMutation = useMutation({
  1868. mutationFn: (mode: 'cooling' | 'heating') => api.setAirductMode(printer.id, mode),
  1869. onMutate: async (mode) => {
  1870. await queryClient.cancelQueries({ queryKey: ['printerStatus', printer.id] });
  1871. const previousStatus = queryClient.getQueryData(['printerStatus', printer.id]);
  1872. queryClient.setQueryData(['printerStatus', printer.id], (old: typeof status) => ({
  1873. ...old,
  1874. airduct_mode: mode === 'cooling' ? 0 : 1,
  1875. }));
  1876. return { previousStatus };
  1877. },
  1878. onError: (error: Error, _, context) => {
  1879. if (context?.previousStatus) {
  1880. queryClient.setQueryData(['printerStatus', printer.id], context.previousStatus);
  1881. }
  1882. showToast(error.message || t('printers.toast.failedToSendCommand'), 'error');
  1883. },
  1884. });
  1885. const bedJogMutation = useMutation({
  1886. mutationFn: ({ distance, force }: { distance: number; force?: boolean }) =>
  1887. api.bedJog(printer.id, distance, force ?? false),
  1888. onError: (error: Error) =>
  1889. showToast(error.message || t('printers.toast.failedToSendCommand'), 'error'),
  1890. });
  1891. const homeAxesMutation = useMutation({
  1892. mutationFn: (axes: 'z' | 'xy' | 'all') => api.homeAxes(printer.id, axes),
  1893. onSuccess: () => {
  1894. // Flip the session-scoped "warned" flag so the next bed-jog click doesn't re-prompt
  1895. // the not-homed modal. The flag is the same one "Move anyway" sets; after a successful
  1896. // auto-home request the printer is (or will shortly be) in a known-homed state, so
  1897. // prompting again in the same session is noise — #1052 follow-up.
  1898. try { sessionStorage.setItem(`bambuddy.bedJog.warned.${printer.id}`, '1'); } catch { /* ignore */ }
  1899. showToast(t('printers.bedJog.homingStarted'));
  1900. },
  1901. onError: (error: Error) =>
  1902. showToast(error.message || t('printers.toast.failedToSendCommand'), 'error'),
  1903. });
  1904. // Plate detection setting mutation
  1905. const plateDetectionMutation = useMutation({
  1906. mutationFn: (enabled: boolean) => api.updatePrinter(printer.id, { plate_detection_enabled: enabled }),
  1907. onSuccess: () => {
  1908. queryClient.invalidateQueries({ queryKey: ['printers'] });
  1909. showToast(plateDetectionMutation.variables ? t('printers.toast.plateCheckEnabled') : t('printers.toast.plateCheckDisabled'));
  1910. },
  1911. onError: (error: Error) => showToast(error.message || t('printers.toast.failedToUpdateSetting'), 'error'),
  1912. });
  1913. // Query for printable objects (for skip functionality)
  1914. // Fetch when printing with 2+ objects OR when modal is open
  1915. const isPrintingWithObjects = (status?.state === 'RUNNING' || status?.state === 'PAUSE') && (status?.printable_objects_count ?? 0) >= 2;
  1916. const { data: objectsData } = useQuery({
  1917. queryKey: ['printableObjects', printer.id],
  1918. queryFn: () => api.getPrintableObjects(printer.id),
  1919. enabled: showSkipObjectsModal || isPrintingWithObjects,
  1920. refetchInterval: showSkipObjectsModal ? 5000 : (isPrintingWithObjects ? 30000 : false), // 5s when modal open, 30s otherwise
  1921. });
  1922. // State for tracking which AMS slot is being refreshed
  1923. const [refreshingSlot, setRefreshingSlot] = useState<{ amsId: number; slotId: number } | null>(null);
  1924. // Track if we've seen the printer enter "busy" state (ams_status_main !== 0)
  1925. const seenBusyStateRef = useRef<boolean>(false);
  1926. // Fallback timeout ref
  1927. const refreshTimeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null);
  1928. // Minimum display time passed
  1929. const minTimePassedRef = useRef<boolean>(false);
  1930. // AMS slot refresh mutation
  1931. const refreshAmsSlotMutation = useMutation({
  1932. mutationFn: ({ amsId, slotId }: { amsId: number; slotId: number }) =>
  1933. api.refreshAmsSlot(printer.id, amsId, slotId),
  1934. onMutate: ({ amsId, slotId }) => {
  1935. // Clear any existing timeout
  1936. if (refreshTimeoutRef.current) {
  1937. clearTimeout(refreshTimeoutRef.current);
  1938. }
  1939. // Reset state
  1940. seenBusyStateRef.current = false;
  1941. minTimePassedRef.current = false;
  1942. setRefreshingSlot({ amsId, slotId });
  1943. // Minimum display time (2 seconds)
  1944. setTimeout(() => {
  1945. minTimePassedRef.current = true;
  1946. }, 2000);
  1947. // Fallback timeout (30 seconds max)
  1948. refreshTimeoutRef.current = setTimeout(() => {
  1949. setRefreshingSlot(null);
  1950. }, 30000);
  1951. },
  1952. onSuccess: (data) => {
  1953. showToast(data.message || t('printers.toast.rfidRereadInitiated'));
  1954. },
  1955. onError: (error: Error) => {
  1956. showToast(error.message || t('printers.toast.failedToRereadRfid'), 'error');
  1957. if (refreshTimeoutRef.current) {
  1958. clearTimeout(refreshTimeoutRef.current);
  1959. }
  1960. setRefreshingSlot(null);
  1961. },
  1962. });
  1963. // AMS load/unload mutations (#891)
  1964. const loadAmsTrayMutation = useMutation({
  1965. mutationFn: ({ trayId }: { trayId: number }) => api.loadAmsTray(printer.id, trayId),
  1966. onSuccess: (data) => {
  1967. showToast(data.message || t('printers.toast.loadInitiated'));
  1968. },
  1969. onError: (error: Error) => {
  1970. showToast(error.message || t('printers.toast.failedToLoad'), 'error');
  1971. },
  1972. });
  1973. const unloadAmsMutation = useMutation({
  1974. mutationFn: () => api.unloadAms(printer.id),
  1975. onSuccess: (data) => {
  1976. showToast(data.message || t('printers.toast.unloadInitiated'));
  1977. },
  1978. onError: (error: Error) => {
  1979. showToast(error.message || t('printers.toast.failedToUnload'), 'error');
  1980. },
  1981. });
  1982. // Plate references state
  1983. const [plateReferences, setPlateReferences] = useState<{
  1984. references: Array<{ index: number; label: string; timestamp: string; has_image: boolean; thumbnail_url: string }>;
  1985. max_references: number;
  1986. } | null>(null);
  1987. const [editingRefLabel, setEditingRefLabel] = useState<{ index: number; label: string } | null>(null);
  1988. // Fetch plate references
  1989. const fetchPlateReferences = async () => {
  1990. try {
  1991. const data = await api.getPlateReferences(printer.id);
  1992. setPlateReferences(data);
  1993. } catch {
  1994. // Ignore errors - references will show as empty
  1995. }
  1996. };
  1997. // Toggle plate detection enabled/disabled
  1998. const handleTogglePlateDetection = () => {
  1999. plateDetectionMutation.mutate(!printer.plate_detection_enabled);
  2000. };
  2001. // Open plate detection management modal (for calibration/references)
  2002. const handleOpenPlateManagement = async () => {
  2003. setIsCheckingPlate(true);
  2004. setPlateCheckResult(null);
  2005. // Auto-turn on light if it's off
  2006. const lightWasOff = status?.chamber_light === false;
  2007. setPlateCheckLightWasOff(lightWasOff);
  2008. if (lightWasOff) {
  2009. await api.setChamberLight(printer.id, true);
  2010. // Wait for light to physically turn on and camera to adjust exposure
  2011. // (MQTT command is async, light takes ~1s to turn on, camera needs time to adjust)
  2012. await new Promise(resolve => setTimeout(resolve, 2500));
  2013. }
  2014. try {
  2015. const result = await api.checkPlateEmpty(printer.id, { includeDebugImage: true });
  2016. setPlateCheckResult(result);
  2017. fetchPlateReferences();
  2018. } catch (error) {
  2019. showToast(error instanceof Error ? error.message : t('printers.toast.failedToCheckPlate'), 'error');
  2020. // Restore light if check failed
  2021. if (lightWasOff) {
  2022. await api.setChamberLight(printer.id, false);
  2023. setPlateCheckLightWasOff(false);
  2024. }
  2025. } finally {
  2026. setIsCheckingPlate(false);
  2027. }
  2028. };
  2029. // Close plate check modal and restore light state
  2030. const closePlateCheckModal = useCallback(async () => {
  2031. setPlateCheckResult(null);
  2032. // Restore light to original state if we turned it on
  2033. if (plateCheckLightWasOff) {
  2034. await api.setChamberLight(printer.id, false);
  2035. setPlateCheckLightWasOff(false);
  2036. }
  2037. }, [plateCheckLightWasOff, printer.id]);
  2038. // Calibrate plate detection handler
  2039. const handleCalibratePlate = async (label?: string) => {
  2040. setIsCalibrating(true);
  2041. try {
  2042. const result = await api.calibratePlateDetection(printer.id, { label });
  2043. if (result.success) {
  2044. showToast(result.message || t('printers.toast.calibrationSaved'), 'success');
  2045. // Refresh references and re-check
  2046. fetchPlateReferences();
  2047. const checkResult = await api.checkPlateEmpty(printer.id, { includeDebugImage: true });
  2048. setPlateCheckResult(checkResult);
  2049. } else {
  2050. showToast(result.message || t('printers.toast.calibrationFailed'), 'error');
  2051. }
  2052. } catch (error) {
  2053. showToast(error instanceof Error ? error.message : t('printers.toast.calibrationFailed'), 'error');
  2054. } finally {
  2055. setIsCalibrating(false);
  2056. }
  2057. };
  2058. // Update reference label
  2059. const handleUpdateRefLabel = async (index: number, label: string) => {
  2060. try {
  2061. await api.updatePlateReferenceLabel(printer.id, index, label);
  2062. setEditingRefLabel(null);
  2063. fetchPlateReferences();
  2064. } catch (error) {
  2065. showToast(error instanceof Error ? error.message : t('printers.toast.failedToUpdateLabel'), 'error');
  2066. }
  2067. };
  2068. // Delete reference
  2069. const handleDeleteRef = async (index: number) => {
  2070. try {
  2071. await api.deletePlateReference(printer.id, index);
  2072. showToast(t('printers.toast.referenceDeleted'), 'success');
  2073. fetchPlateReferences();
  2074. // Re-check to update counts
  2075. const checkResult = await api.checkPlateEmpty(printer.id, { includeDebugImage: true });
  2076. setPlateCheckResult(checkResult);
  2077. } catch (error) {
  2078. showToast(error instanceof Error ? error.message : t('printers.toast.failedToDeleteReference'), 'error');
  2079. }
  2080. };
  2081. // Save ROI settings
  2082. const handleSaveRoi = async () => {
  2083. if (!editingRoi) return;
  2084. setIsSavingRoi(true);
  2085. try {
  2086. await api.updatePrinter(printer.id, { plate_detection_roi: editingRoi });
  2087. showToast(t('printers.toast.detectionAreaSaved'), 'success');
  2088. setEditingRoi(null);
  2089. // Re-check to see new ROI in action
  2090. const checkResult = await api.checkPlateEmpty(printer.id, { includeDebugImage: true });
  2091. setPlateCheckResult(checkResult);
  2092. } catch (error) {
  2093. showToast(error instanceof Error ? error.message : t('printers.toast.failedToSaveDetectionArea'), 'error');
  2094. } finally {
  2095. setIsSavingRoi(false);
  2096. }
  2097. };
  2098. // Close plate check modal on Escape key
  2099. useEffect(() => {
  2100. const handleEscape = (e: KeyboardEvent) => {
  2101. if (e.key === 'Escape' && plateCheckResult) {
  2102. closePlateCheckModal();
  2103. }
  2104. };
  2105. window.addEventListener('keydown', handleEscape);
  2106. return () => window.removeEventListener('keydown', handleEscape);
  2107. }, [plateCheckResult, closePlateCheckModal]);
  2108. // Watch ams_status_main to detect when RFID read completes
  2109. // ams_status_main: 0=idle, 2=rfid_identifying
  2110. const deferredClearRef = useRef<ReturnType<typeof setTimeout> | null>(null);
  2111. useEffect(() => {
  2112. if (!refreshingSlot) return;
  2113. const amsStatus = status?.ams_status_main ?? 0;
  2114. // Track when we see non-idle state (printer is working)
  2115. if (amsStatus !== 0) {
  2116. seenBusyStateRef.current = true;
  2117. // Cancel any deferred clear since we're back to busy
  2118. if (deferredClearRef.current) {
  2119. clearTimeout(deferredClearRef.current);
  2120. deferredClearRef.current = null;
  2121. }
  2122. }
  2123. // When we've seen busy and now idle, clear (with min time check)
  2124. if (seenBusyStateRef.current && amsStatus === 0) {
  2125. if (minTimePassedRef.current) {
  2126. // Min time passed - clear now
  2127. if (refreshTimeoutRef.current) {
  2128. clearTimeout(refreshTimeoutRef.current);
  2129. }
  2130. setRefreshingSlot(null);
  2131. } else {
  2132. // Schedule clear after min time (2 seconds from start)
  2133. if (!deferredClearRef.current) {
  2134. deferredClearRef.current = setTimeout(() => {
  2135. if (refreshTimeoutRef.current) {
  2136. clearTimeout(refreshTimeoutRef.current);
  2137. }
  2138. setRefreshingSlot(null);
  2139. }, 2000);
  2140. }
  2141. }
  2142. }
  2143. return () => {
  2144. if (deferredClearRef.current) {
  2145. clearTimeout(deferredClearRef.current);
  2146. }
  2147. };
  2148. }, [status?.ams_status_main, refreshingSlot]);
  2149. // State for AMS slot menu
  2150. const [amsSlotMenu, setAmsSlotMenu] = useState<{ amsId: number; slotId: number } | null>(null);
  2151. if (shouldHide) {
  2152. return null;
  2153. }
  2154. // Size-based styling helpers
  2155. const getImageSize = () => {
  2156. switch (cardSize) {
  2157. case 1: return 'w-10 h-10';
  2158. case 2: return 'w-14 h-14';
  2159. case 3: return 'w-16 h-16';
  2160. case 4: return 'w-20 h-20';
  2161. default: return 'w-14 h-14';
  2162. }
  2163. };
  2164. const getTitleSize = () => {
  2165. switch (cardSize) {
  2166. case 1: return 'text-base truncate';
  2167. case 2: return 'text-lg';
  2168. case 3: return 'text-xl';
  2169. case 4: return 'text-2xl';
  2170. default: return 'text-lg';
  2171. }
  2172. };
  2173. const getSpacing = () => {
  2174. switch (cardSize) {
  2175. case 1: return 'mb-2';
  2176. case 2: return 'mb-4';
  2177. case 3: return 'mb-5';
  2178. case 4: return 'mb-6';
  2179. default: return 'mb-4';
  2180. }
  2181. };
  2182. const canDrop = isConnected && status?.state !== 'RUNNING' && status?.state !== 'PAUSE' && hasPermission('printers:control');
  2183. const handleCardDragEnter = (e: React.DragEvent) => {
  2184. e.preventDefault();
  2185. dragCounterRef.current++;
  2186. if (dragCounterRef.current === 1) setIsDraggingFile(true);
  2187. };
  2188. const handleCardDragOver = (e: React.DragEvent) => {
  2189. e.preventDefault();
  2190. e.dataTransfer.dropEffect = canDrop ? 'copy' : 'none';
  2191. };
  2192. const handleCardDragLeave = (e: React.DragEvent) => {
  2193. e.preventDefault();
  2194. dragCounterRef.current--;
  2195. if (dragCounterRef.current === 0) setIsDraggingFile(false);
  2196. };
  2197. const handleCardDrop = async (e: React.DragEvent) => {
  2198. e.preventDefault();
  2199. dragCounterRef.current = 0;
  2200. setIsDraggingFile(false);
  2201. if (!canDrop) return;
  2202. const droppedFiles = Array.from(e.dataTransfer.files);
  2203. const file = droppedFiles[0];
  2204. if (!file) return;
  2205. // Only accept sliced/printable files (.gcode, .gcode.3mf, etc.)
  2206. const lower = file.name.toLowerCase();
  2207. if (!lower.endsWith('.gcode') && !lower.includes('.gcode.')) {
  2208. showToast(t('printers.dropNotPrintable', 'Only .gcode and .gcode.3mf files can be printed'), 'error');
  2209. return;
  2210. }
  2211. setIsDropUploading(true);
  2212. try {
  2213. const result = await api.uploadLibraryFile(file, null);
  2214. // Check printer compatibility if sliced_for_model is available in metadata
  2215. const slicedFor = (result.metadata as Record<string, unknown>)?.sliced_for_model as string | undefined;
  2216. const printerModel = mapModelCode(printer.model);
  2217. if (slicedFor && printerModel && slicedFor.toLowerCase() !== printerModel.toLowerCase()) {
  2218. await api.deleteLibraryFile(result.id).catch(() => {});
  2219. showToast(
  2220. t('printers.incompatibleFile', 'This file was sliced for {{slicedFor}}, but this printer is a {{printerModel}}', { slicedFor, printerModel }),
  2221. 'error'
  2222. );
  2223. return;
  2224. }
  2225. setPrintAfterUpload({ id: result.id, filename: result.filename });
  2226. } catch {
  2227. showToast(t('common.uploadFailed', 'Upload failed'), 'error');
  2228. } finally {
  2229. setIsDropUploading(false);
  2230. }
  2231. };
  2232. return (
  2233. <Card
  2234. className={`relative ${isSelected ? 'ring-2 ring-bambu-green' : ''} ${selectionMode ? 'cursor-pointer' : ''}`}
  2235. onDragEnter={handleCardDragEnter}
  2236. onDragOver={handleCardDragOver}
  2237. onDragLeave={handleCardDragLeave}
  2238. onDrop={handleCardDrop}
  2239. >
  2240. {/* Selection mode click overlay — captures all clicks, preventing nested interactions */}
  2241. {selectionMode && (
  2242. <div
  2243. className="absolute inset-0 z-20 flex items-start p-2"
  2244. onClick={(e) => { e.stopPropagation(); onToggleSelect?.(printer.id); }}
  2245. >
  2246. {isSelected ? (
  2247. <CheckSquare className="w-5 h-5 text-bambu-green" />
  2248. ) : (
  2249. <Square className="w-5 h-5 text-bambu-gray" />
  2250. )}
  2251. </div>
  2252. )}
  2253. {/* Drop zone overlay */}
  2254. {(isDraggingFile || isDropUploading) && (
  2255. <div
  2256. className={`absolute inset-0 z-10 rounded-xl border-2 border-dashed flex items-center justify-center transition-colors ${
  2257. isDropUploading
  2258. ? 'bg-bambu-green/10 border-bambu-green/50'
  2259. : canDrop
  2260. ? 'bg-bambu-green/10 border-bambu-green'
  2261. : 'bg-red-500/10 border-red-500/50'
  2262. }`}
  2263. >
  2264. <div className="text-center">
  2265. {isDropUploading ? (
  2266. <>
  2267. <Loader2 className="w-8 h-8 mx-auto mb-2 text-bambu-green animate-spin" />
  2268. <p className="text-sm font-medium text-bambu-green">{t('common.uploading', 'Uploading...')}</p>
  2269. </>
  2270. ) : canDrop ? (
  2271. <>
  2272. <PrinterIcon className="w-8 h-8 mx-auto mb-2 text-bambu-green" />
  2273. <p className="text-sm font-medium text-bambu-green">{t('printers.dropToPrint', 'Drop to print')}</p>
  2274. </>
  2275. ) : (
  2276. <>
  2277. <X className="w-8 h-8 mx-auto mb-2 text-red-400" />
  2278. <p className="text-sm font-medium text-red-400">{t('printers.cannotPrint', 'Printer busy')}</p>
  2279. </>
  2280. )}
  2281. </div>
  2282. </div>
  2283. )}
  2284. <CardContent className={cardSize >= 3 ? 'p-5' : ''}>
  2285. {/* Header */}
  2286. <div className={getSpacing()}>
  2287. {/* Top row: Image, Name, Menu */}
  2288. <div className="flex items-start justify-between gap-2">
  2289. <div className="flex items-center gap-3 min-w-0 flex-1">
  2290. {/* Printer Model Image */}
  2291. <img
  2292. src={getPrinterImage(printer.model)}
  2293. alt={printer.model || t('common.printer')}
  2294. className={`object-contain rounded-lg bg-bambu-dark flex-shrink-0 ${getImageSize()}`}
  2295. />
  2296. <div className="min-w-0 flex-1">
  2297. <div className="flex items-center gap-2">
  2298. <h3 className={`font-semibold text-white ${getTitleSize()}`}>{printer.name}</h3>
  2299. {/* Connection indicator dot for compact mode */}
  2300. {viewMode === 'compact' && (() => {
  2301. const hmsErrors = status?.connected && status.hms_errors ? filterKnownHMSErrors(status.hms_errors) : [];
  2302. const hasSevere = hmsErrors.some(e => e.severity <= 2);
  2303. const hasWarning = hmsErrors.length > 0;
  2304. const pipColor = !status?.connected
  2305. ? 'bg-status-error'
  2306. : hasSevere
  2307. ? 'bg-status-error'
  2308. : hasWarning
  2309. ? 'bg-status-warning'
  2310. : 'bg-status-ok';
  2311. const pipTitle = !status?.connected
  2312. ? t('printers.connection.offline')
  2313. : hasWarning
  2314. ? `${hmsErrors.length} HMS ${hmsErrors.length === 1 ? 'error' : 'errors'}`
  2315. : t('printers.connection.connected');
  2316. return (
  2317. <div
  2318. className={`w-2 h-2 rounded-full flex-shrink-0 ${pipColor}`}
  2319. title={pipTitle}
  2320. />
  2321. );
  2322. })()}
  2323. </div>
  2324. <p className="text-sm text-bambu-gray">
  2325. {printer.model || 'Unknown Model'}
  2326. {/* Nozzle Info - only in expanded */}
  2327. {viewMode === 'expanded' && status?.nozzles && status.nozzles[0]?.nozzle_diameter && (
  2328. <span className="ml-1.5 text-bambu-gray" title={status.nozzles[0].nozzle_type || 'Nozzle'}>
  2329. • {status.nozzles[0].nozzle_diameter}mm
  2330. </span>
  2331. )}
  2332. {viewMode === 'expanded' && maintenanceInfo && maintenanceInfo.total_print_hours > 0 && (
  2333. <span className="ml-2 text-bambu-gray">
  2334. <Clock className="w-3 h-3 inline-block mr-1" />
  2335. {Math.round(maintenanceInfo.total_print_hours)}h
  2336. </span>
  2337. )}
  2338. </p>
  2339. </div>
  2340. </div>
  2341. {/* Menu button */}
  2342. <div className="relative flex-shrink-0">
  2343. <Button
  2344. variant="ghost"
  2345. size="sm"
  2346. onClick={() => setShowMenu(!showMenu)}
  2347. >
  2348. <MoreVertical className="w-4 h-4" />
  2349. </Button>
  2350. {showMenu && (
  2351. <div className="absolute right-0 mt-2 w-48 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg shadow-lg z-20">
  2352. <button
  2353. className={`w-full px-4 py-2 text-left text-sm flex items-center gap-2 ${
  2354. hasPermission('printers:update')
  2355. ? 'hover:bg-bambu-dark-tertiary'
  2356. : 'opacity-50 cursor-not-allowed'
  2357. }`}
  2358. onClick={() => {
  2359. if (!hasPermission('printers:update')) return;
  2360. setShowEditModal(true);
  2361. setShowMenu(false);
  2362. }}
  2363. title={!hasPermission('printers:update') ? t('printers.permission.noEdit') : undefined}
  2364. >
  2365. <Pencil className="w-4 h-4" />
  2366. {t('common.edit')}
  2367. </button>
  2368. <button
  2369. className="w-full px-4 py-2 text-left text-sm hover:bg-bambu-dark-tertiary flex items-center gap-2"
  2370. onClick={() => {
  2371. setShowPrinterInfo(true);
  2372. setShowMenu(false);
  2373. }}
  2374. >
  2375. <Info className="w-4 h-4" />
  2376. {t('printers.printerInformation')}
  2377. </button>
  2378. <button
  2379. className="w-full px-4 py-2 text-left text-sm hover:bg-bambu-dark-tertiary flex items-center gap-2"
  2380. onClick={() => {
  2381. connectMutation.mutate();
  2382. setShowMenu(false);
  2383. }}
  2384. >
  2385. <RefreshCw className="w-4 h-4" />
  2386. {t('printers.reconnect')}
  2387. </button>
  2388. <button
  2389. className="w-full px-4 py-2 text-left text-sm hover:bg-bambu-dark-tertiary flex items-center gap-2 disabled:opacity-50"
  2390. disabled={forceRefreshMutation.isPending}
  2391. onClick={() => {
  2392. forceRefreshMutation.mutate();
  2393. setShowMenu(false);
  2394. }}
  2395. >
  2396. <RotateCw className={`w-4 h-4 ${forceRefreshMutation.isPending ? 'animate-spin' : ''}`} />
  2397. {t('printers.forceRefresh')}
  2398. </button>
  2399. <button
  2400. className="w-full px-4 py-2 text-left text-sm hover:bg-bambu-dark-tertiary flex items-center gap-2"
  2401. onClick={() => {
  2402. setShowMQTTDebug(true);
  2403. setShowMenu(false);
  2404. }}
  2405. >
  2406. <Terminal className="w-4 h-4" />
  2407. {t('printers.mqttDebug')}
  2408. </button>
  2409. <button
  2410. className="w-full px-4 py-2 text-left text-sm hover:bg-bambu-dark-tertiary flex items-center gap-2"
  2411. onClick={() => {
  2412. setShowDiagnostic(true);
  2413. setShowMenu(false);
  2414. }}
  2415. >
  2416. <Stethoscope className="w-4 h-4" />
  2417. {t('diagnostic.runButton')}
  2418. </button>
  2419. <button
  2420. className={`w-full px-4 py-2 text-left text-sm flex items-center gap-2 ${
  2421. hasPermission('printers:delete')
  2422. ? 'text-red-400 hover:bg-bambu-dark-tertiary'
  2423. : 'text-red-400/50 cursor-not-allowed'
  2424. }`}
  2425. onClick={() => {
  2426. if (!hasPermission('printers:delete')) return;
  2427. setShowDeleteConfirm(true);
  2428. setShowMenu(false);
  2429. }}
  2430. title={!hasPermission('printers:delete') ? t('printers.permission.noDelete') : undefined}
  2431. >
  2432. <Trash2 className="w-4 h-4" />
  2433. {t('common.delete')}
  2434. </button>
  2435. </div>
  2436. )}
  2437. </div>
  2438. </div>
  2439. {/* Badges row - only in expanded mode */}
  2440. {viewMode === 'expanded' && (
  2441. <div className="flex flex-wrap items-center gap-2 mt-2">
  2442. {/* Connection status badge */}
  2443. <span
  2444. className={`flex items-center gap-1.5 px-2 py-1 rounded-full text-xs ${
  2445. status?.connected
  2446. ? 'bg-status-ok/20 text-status-ok'
  2447. : 'bg-status-error/20 text-status-error'
  2448. }`}
  2449. >
  2450. {status?.connected ? (
  2451. <Link className="w-3 h-3" />
  2452. ) : (
  2453. <Unlink className="w-3 h-3" />
  2454. )}
  2455. {status?.connected ? t('printers.connection.connected') : t('printers.connection.offline')}
  2456. </span>
  2457. {/* Run connection diagnostic — offered when the printer is offline */}
  2458. {!status?.connected && (
  2459. <button
  2460. onClick={() => setShowDiagnostic(true)}
  2461. className="flex items-center gap-1 px-2 py-1 rounded-full text-xs cursor-pointer bg-bambu-dark-tertiary text-bambu-gray hover:text-white transition-colors"
  2462. title={t('diagnostic.runButton')}
  2463. >
  2464. <Stethoscope className="w-3 h-3" />
  2465. {t('diagnostic.runButton')}
  2466. </button>
  2467. )}
  2468. {/* Network connection indicator */}
  2469. {status?.connected && status?.wired_network && (
  2470. <span
  2471. className="flex items-center gap-1 px-2 py-1 rounded-full text-xs bg-status-ok/20 text-status-ok"
  2472. title={t('printers.connection.ethernet', 'Ethernet')}
  2473. >
  2474. <Cable className="w-3 h-3" />
  2475. {t('printers.connection.ethernet', 'Ethernet')}
  2476. </span>
  2477. )}
  2478. {/* WiFi signal indicator */}
  2479. {status?.connected && !status?.wired_network && wifiSignal != null && (
  2480. <span
  2481. className={`flex items-center gap-1 px-2 py-1 rounded-full text-xs ${
  2482. wifiSignal >= -50
  2483. ? 'bg-status-ok/20 text-status-ok'
  2484. : wifiSignal >= -60
  2485. ? 'bg-status-ok/20 text-status-ok'
  2486. : wifiSignal >= -70
  2487. ? 'bg-status-warning/20 text-status-warning'
  2488. : wifiSignal >= -80
  2489. ? 'bg-orange-500/20 text-orange-600'
  2490. : 'bg-status-error/20 text-status-error'
  2491. }`}
  2492. title={`WiFi: ${wifiSignal} dBm - ${t(getWifiStrength(wifiSignal).labelKey)}`}
  2493. >
  2494. <Signal className="w-3 h-3" />
  2495. {wifiSignal}dBm
  2496. </span>
  2497. )}
  2498. {/* HMS Status Indicator */}
  2499. {status?.connected && (() => {
  2500. const knownErrors = status.hms_errors ? filterKnownHMSErrors(status.hms_errors) : [];
  2501. return (
  2502. <button
  2503. onClick={() => setShowHMSModal(true)}
  2504. className={`flex items-center gap-1 px-2 py-1 rounded-full text-xs cursor-pointer hover:opacity-80 transition-opacity ${
  2505. knownErrors.length > 0
  2506. ? knownErrors.some(e => e.severity <= 2)
  2507. ? 'bg-status-error/20 text-status-error'
  2508. : 'bg-status-warning/20 text-status-warning'
  2509. : 'bg-status-ok/20 text-status-ok'
  2510. }`}
  2511. title={t('printers.clickToViewHmsErrors')}
  2512. >
  2513. <AlertTriangle className="w-3 h-3" />
  2514. {knownErrors.length > 0 ? knownErrors.length : 'OK'}
  2515. </button>
  2516. );
  2517. })()}
  2518. {/* Maintenance Status Indicator */}
  2519. {maintenanceInfo && (
  2520. <button
  2521. onClick={() => navigate('/maintenance')}
  2522. className={`flex items-center gap-1 px-2 py-1 rounded-full text-xs cursor-pointer hover:opacity-80 transition-opacity ${
  2523. maintenanceInfo.due_count > 0
  2524. ? 'bg-status-error/20 text-status-error'
  2525. : maintenanceInfo.warning_count > 0
  2526. ? 'bg-status-warning/20 text-status-warning'
  2527. : 'bg-status-ok/20 text-status-ok'
  2528. }`}
  2529. title={
  2530. maintenanceInfo.due_count > 0 || maintenanceInfo.warning_count > 0
  2531. ? `${maintenanceInfo.due_count > 0 ? `${maintenanceInfo.due_count} maintenance due` : ''}${maintenanceInfo.due_count > 0 && maintenanceInfo.warning_count > 0 ? ', ' : ''}${maintenanceInfo.warning_count > 0 ? `${maintenanceInfo.warning_count} due soon` : ''} - Click to view`
  2532. : t('printers.maintenanceUpToDate')
  2533. }
  2534. >
  2535. <Wrench className="w-3 h-3" />
  2536. {maintenanceInfo.due_count > 0 || maintenanceInfo.warning_count > 0
  2537. ? maintenanceInfo.due_count + maintenanceInfo.warning_count
  2538. : 'OK'}
  2539. </button>
  2540. )}
  2541. {/* Queue Count Badge */}
  2542. {queueCount > 0 && (
  2543. <button
  2544. onClick={() => navigate('/queue')}
  2545. className="flex items-center gap-1 px-2 py-1 rounded-full text-xs bg-indigo-500/20 text-indigo-400 hover:opacity-80 transition-opacity"
  2546. title={t('printers.queue.inQueue', { count: queueCount })}
  2547. >
  2548. <Layers className="w-3 h-3" />
  2549. {queueCount}
  2550. </button>
  2551. )}
  2552. {/* Firmware Version Badge */}
  2553. {checkPrinterFirmware && firmwareInfo?.current_version && firmwareInfo?.latest_version ? (
  2554. <button
  2555. onClick={() => setShowFirmwareModal(true)}
  2556. className={`flex items-center gap-1 px-2 py-1 rounded-full text-xs hover:opacity-80 transition-opacity ${
  2557. firmwareInfo.update_available
  2558. ? 'bg-orange-500/20 text-orange-400'
  2559. : 'bg-status-ok/20 text-status-ok'
  2560. }`}
  2561. title={
  2562. firmwareInfo.update_available
  2563. ? t('printers.firmwareUpdateAvailable', { current: firmwareInfo.current_version, latest: firmwareInfo.latest_version })
  2564. : t('printers.firmwareUpToDate', { version: firmwareInfo.current_version })
  2565. }
  2566. >
  2567. {firmwareInfo.update_available ? <Download className="w-3 h-3" /> : <CheckCircle className="w-3 h-3" />}
  2568. {firmwareInfo.current_version}
  2569. </button>
  2570. ) : status?.firmware_version ? (
  2571. <span className="flex items-center gap-1 px-2 py-1 rounded-full text-xs bg-bambu-dark-tertiary/50 text-bambu-gray">
  2572. {status.firmware_version}
  2573. </span>
  2574. ) : null}
  2575. {/* Enclosure Door Badge (X1/X2D/P1S/P2S/H2*) */}
  2576. {status?.connected && ['X1C', 'X1', 'X1E', 'X2D', 'P1S', 'P1P', 'P2S', 'H2D', 'H2D Pro', 'H2C', 'H2S'].includes(printer.model ?? '') && (
  2577. <span
  2578. className={`flex items-center px-2 py-1 rounded-full text-xs ${
  2579. status.door_open
  2580. ? 'bg-yellow-500/20 text-yellow-400'
  2581. : 'bg-status-ok/20 text-status-ok'
  2582. }`}
  2583. title={status.door_open ? t('printers.door.open') : t('printers.door.closed')}
  2584. >
  2585. {status.door_open ? <DoorOpen className="w-3 h-3" /> : <DoorClosed className="w-3 h-3" />}
  2586. </span>
  2587. )}
  2588. </div>
  2589. )}
  2590. </div>
  2591. {/* Delete Confirmation */}
  2592. {showDeleteConfirm && (
  2593. <div className="fixed inset-0 bg-black/50 flex items-center justify-center z-50">
  2594. <Card className="w-full max-w-md mx-4">
  2595. <CardContent>
  2596. <div className="flex items-start gap-3 mb-4">
  2597. <div className="p-2 rounded-full bg-red-500/20">
  2598. <AlertTriangle className="w-5 h-5 text-red-400" />
  2599. </div>
  2600. <div>
  2601. <h3 className="text-lg font-semibold text-white">{t('printers.confirm.deleteTitle')}</h3>
  2602. <p className="text-sm text-bambu-gray mt-1">
  2603. {t('printers.confirm.deleteMessage', { name: printer.name })}
  2604. </p>
  2605. </div>
  2606. </div>
  2607. <div className="bg-bambu-dark rounded-lg p-3 mb-4">
  2608. <label className="flex items-start gap-3 cursor-pointer">
  2609. <input
  2610. type="checkbox"
  2611. checked={deleteArchives}
  2612. onChange={(e) => setDeleteArchives(e.target.checked)}
  2613. className="mt-0.5 w-4 h-4 rounded border-bambu-gray bg-bambu-dark-secondary text-bambu-green focus:ring-bambu-green focus:ring-offset-0"
  2614. />
  2615. <div>
  2616. <span className="text-sm text-white">{t('printers.deleteArchives')}</span>
  2617. <p className="text-xs text-bambu-gray mt-0.5">
  2618. {deleteArchives
  2619. ? t('printers.confirm.deleteArchivesNote')
  2620. : t('printers.confirm.keepArchivesNote')}
  2621. </p>
  2622. </div>
  2623. </label>
  2624. </div>
  2625. <div className="flex justify-end gap-2">
  2626. <Button
  2627. variant="secondary"
  2628. onClick={() => {
  2629. setShowDeleteConfirm(false);
  2630. setDeleteArchives(true);
  2631. }}
  2632. >
  2633. {t('common.cancel')}
  2634. </Button>
  2635. <Button
  2636. variant="danger"
  2637. onClick={() => {
  2638. deleteMutation.mutate({ deleteArchives });
  2639. setShowDeleteConfirm(false);
  2640. setDeleteArchives(true);
  2641. }}
  2642. >
  2643. Delete
  2644. </Button>
  2645. </div>
  2646. </CardContent>
  2647. </Card>
  2648. </div>
  2649. )}
  2650. {/* Status */}
  2651. {status?.connected && (
  2652. <>
  2653. {/* Compact: Simple status bar */}
  2654. {viewMode === 'compact' ? (
  2655. <div className="mt-2">
  2656. {(status.state === 'RUNNING' || status.state === 'PAUSE') ? (
  2657. <div className="flex items-center gap-2">
  2658. <div className="flex-1 bg-bambu-dark-tertiary rounded-full h-1.5">
  2659. <div
  2660. className={`${status.state === 'PAUSE' ? 'bg-status-warning' : 'bg-bambu-green'} h-1.5 rounded-full transition-all`}
  2661. style={{ width: `${status.progress || 0}%` }}
  2662. />
  2663. </div>
  2664. <div className="flex flex-shrink-0 items-center gap-1.5">
  2665. <span className="text-xs text-white">{Math.round(status.progress || 0)}%</span>
  2666. {plateStatusPill}
  2667. </div>
  2668. </div>
  2669. ) : (
  2670. <div className="flex items-center justify-between gap-2">
  2671. <div className="min-w-0 flex-1 flex items-center gap-1.5">
  2672. <p className="min-w-0 truncate text-xs text-bambu-gray">{getStatusDisplay(status.state, status.stg_cur_name)}</p>
  2673. {plateStatusPill}
  2674. </div>
  2675. {showClearPlateButton && (
  2676. <button
  2677. type="button"
  2678. onClick={() => clearPlateMutation.mutate()}
  2679. disabled={clearPlateMutation.isPending || !hasPermission('printers:clear_plate')}
  2680. aria-label={t('printers.plateStatus.markCleared')}
  2681. className="inline-flex h-5 w-5 flex-shrink-0 items-center justify-center rounded-full bg-yellow-500/20 border border-yellow-400/40 text-yellow-400 hover:bg-yellow-500/30 transition-colors disabled:opacity-50"
  2682. title={!hasPermission('printers:clear_plate') ? t('printers.permission.noControl') : t('printers.plateStatus.markCleared')}
  2683. >
  2684. {clearPlateMutation.isPending ? (
  2685. <Loader2 className="w-3 h-3 animate-spin" />
  2686. ) : (
  2687. <PlateClearedIcon className="w-3 h-3" />
  2688. )}
  2689. </button>
  2690. )}
  2691. </div>
  2692. )}
  2693. </div>
  2694. ) : (
  2695. /* Expanded: Full status section */
  2696. <>
  2697. {/* Current Print or Idle Placeholder */}
  2698. <div className="mb-4 p-3 bg-bambu-dark rounded-lg relative">
  2699. {/* Skip Objects button - top right corner, always visible */}
  2700. <button
  2701. onClick={() => setShowSkipObjectsModal(true)}
  2702. disabled={!(status.state === 'RUNNING' || status.state === 'PAUSE') || (status.printable_objects_count ?? 0) < 2 || !hasPermission('printers:control')}
  2703. className={`absolute top-2 right-2 p-1.5 rounded transition-colors z-10 ${
  2704. (status.state === 'RUNNING' || status.state === 'PAUSE') && (status.printable_objects_count ?? 0) >= 2 && hasPermission('printers:control')
  2705. ? 'text-bambu-gray hover:text-white hover:bg-white/10'
  2706. : 'text-bambu-gray/30 cursor-not-allowed'
  2707. }`}
  2708. title={
  2709. !hasPermission('printers:control')
  2710. ? t('printers.permission.noControl')
  2711. : !(status.state === 'RUNNING' || status.state === 'PAUSE')
  2712. ? t('printers.skipObjects.onlyWhilePrinting')
  2713. : (status.printable_objects_count ?? 0) >= 2
  2714. ? t('printers.skipObjects.tooltip')
  2715. : t('printers.skipObjects.requiresMultiple')
  2716. }
  2717. >
  2718. <SkipObjectsIcon className="w-4 h-4" />
  2719. {/* Badge showing skipped count */}
  2720. {objectsData && objectsData.skipped_count > 0 && (
  2721. <span className="absolute -top-1 -right-1 min-w-[16px] h-4 px-1 flex items-center justify-center text-[10px] font-bold bg-red-500 text-white rounded-full">
  2722. {objectsData.skipped_count}
  2723. </span>
  2724. )}
  2725. </button>
  2726. <div className="flex gap-3">
  2727. {/* Cover Image */}
  2728. <CoverImage
  2729. url={(status.state === 'RUNNING' || status.state === 'PAUSE') ? status.cover_url : null}
  2730. printName={(status.state === 'RUNNING' || status.state === 'PAUSE') ? (formatPrintName(status.subtask_name || status.current_print || null, status.gcode_file, t, activePlateLabel) || undefined) : undefined}
  2731. />
  2732. {/* Print Info */}
  2733. <div className="flex-1 min-w-0">
  2734. {status.current_print && (status.state === 'RUNNING' || status.state === 'PAUSE') ? (
  2735. <>
  2736. <div className="mb-1 flex items-center gap-2">
  2737. <p className="text-sm text-bambu-gray">{getStatusDisplay(status.state, status.stg_cur_name)}</p>
  2738. {plateStatusPill}
  2739. </div>
  2740. <p className="text-white text-sm mb-2 truncate">
  2741. {formatPrintName(status.subtask_name || status.current_print || null, status.gcode_file, t, activePlateLabel)}
  2742. </p>
  2743. <div className="flex items-center justify-between text-sm">
  2744. <div className="flex-1 bg-bambu-dark-tertiary rounded-full h-2 mr-3">
  2745. <div
  2746. className={`${status.state === 'PAUSE' ? 'bg-status-warning' : 'bg-bambu-green'} h-2 rounded-full transition-all`}
  2747. style={{ width: `${status.progress || 0}%` }}
  2748. />
  2749. </div>
  2750. <span className="text-white">{Math.round(status.progress || 0)}%</span>
  2751. </div>
  2752. <div className="flex items-center gap-3 mt-2 text-xs text-bambu-gray">
  2753. {status.remaining_time != null && status.remaining_time > 0 && (
  2754. <>
  2755. <span className="flex items-center gap-1">
  2756. <Clock className="w-3 h-3" />
  2757. {formatDuration(status.remaining_time * 60)}
  2758. </span>
  2759. <span className="text-bambu-green font-medium" title={t('printers.estimatedCompletion')}>
  2760. ETA {formatETA(status.remaining_time, timeFormat, t)}
  2761. </span>
  2762. </>
  2763. )}
  2764. {status.layer_num != null && status.total_layers != null && status.total_layers > 0 && (
  2765. <span className="flex items-center gap-1">
  2766. <Layers className="w-3 h-3" />
  2767. {status.layer_num}/{status.total_layers}
  2768. </span>
  2769. )}
  2770. {currentPrintUser && (
  2771. <span className="flex items-center gap-1" title={`Started by ${currentPrintUser}`}>
  2772. <User className="w-3 h-3" />
  2773. {currentPrintUser}
  2774. </span>
  2775. )}
  2776. </div>
  2777. </>
  2778. ) : (
  2779. <>
  2780. <p className="text-sm text-bambu-gray mb-1">{t('printers.sort.status')}</p>
  2781. <div className="mb-2 flex items-center gap-2">
  2782. <p className="text-white text-sm">
  2783. {getStatusDisplay(status.state, status.stg_cur_name)}
  2784. </p>
  2785. {plateStatusPill}
  2786. </div>
  2787. <div className="flex items-center justify-between text-sm">
  2788. <div className="flex-1 bg-bambu-dark-tertiary rounded-full h-2 mr-3">
  2789. <div className="bg-bambu-dark-tertiary h-2 rounded-full" />
  2790. </div>
  2791. <span className="text-bambu-gray">—</span>
  2792. </div>
  2793. {lastPrint ? (
  2794. <p className="text-xs text-bambu-gray mt-2 truncate" title={lastPrint.print_name || lastPrint.filename}>
  2795. Last: {lastPrint.print_name || lastPrint.filename}
  2796. {lastPrint.completed_at && (
  2797. <span className="ml-1 text-bambu-gray/60">
  2798. • {formatDateOnly(lastPrint.completed_at, { month: 'short', day: 'numeric' })}
  2799. </span>
  2800. )}
  2801. </p>
  2802. ) : (
  2803. <p className="text-xs text-bambu-gray mt-2">{t('printers.readyToPrint')}</p>
  2804. )}
  2805. </>
  2806. )}
  2807. </div>
  2808. </div>
  2809. </div>
  2810. {/* Queue Widget - always visible when there are pending items */}
  2811. <PrinterQueueWidget printerId={printer.id} printerModel={printer.model} loadedFilamentTypes={loadedFilamentTypes} loadedFilaments={loadedFilaments} />
  2812. </>
  2813. )}
  2814. {/* Temperatures */}
  2815. {status.temperatures && viewMode === 'expanded' && (() => {
  2816. // Use actual heater states from MQTT stream
  2817. const nozzleHeating = status.temperatures.nozzle_heating || status.temperatures.nozzle_2_heating || false;
  2818. const bedHeating = status.temperatures.bed_heating || false;
  2819. const chamberHeating = status.temperatures.chamber_heating || false;
  2820. const isDualNozzle = printer.nozzle_count === 2 || status.temperatures.nozzle_2 !== undefined;
  2821. // active_extruder: 0=right, 1=left
  2822. const activeNozzle = status.active_extruder === 1 ? 'L' : 'R';
  2823. // Extended nozzle data from nozzle_rack (H2 series: wear, serial, max_temp, etc.)
  2824. // nozzle_rack id 0 = extruder 0 = RIGHT, id 1 = extruder 1 = LEFT
  2825. const leftNozzleSlot = status.nozzle_rack?.find(s => s.id === 1);
  2826. const rightNozzleSlot = status.nozzle_rack?.find(s => s.id === 0);
  2827. // Single-nozzle models (H2D, H2C): use the primary nozzle (id 0)
  2828. const singleNozzleSlot = rightNozzleSlot || leftNozzleSlot;
  2829. return (
  2830. <div className="flex items-stretch gap-1.5 flex-wrap">
  2831. {/* Nozzle temp - combined for dual nozzle */}
  2832. <div className="text-center px-2 py-1.5 bg-bambu-dark rounded-lg flex-1 flex flex-col justify-center items-center">
  2833. <HeaterThermometer className="w-3.5 h-3.5 mb-0.5" color="text-orange-400" isHeating={nozzleHeating} />
  2834. {status.temperatures.nozzle_2 !== undefined ? (
  2835. <>
  2836. <p className="text-[9px] text-bambu-gray">L / R</p>
  2837. <p className="text-[11px] text-white">
  2838. {Math.round(status.temperatures.nozzle || 0)}° / {Math.round(status.temperatures.nozzle_2 || 0)}°
  2839. </p>
  2840. </>
  2841. ) : singleNozzleSlot ? (
  2842. <NozzleSlotHoverCard slot={singleNozzleSlot} index={0} activeStatus filamentName={singleNozzleSlot.filament_id ? filamentInfo?.[singleNozzleSlot.filament_id]?.name : undefined}>
  2843. <div className="cursor-default">
  2844. <p className="text-[9px] text-bambu-gray">{t('printers.temperatures.nozzle')}</p>
  2845. <p className="text-[11px] text-white">
  2846. {Math.round(status.temperatures.nozzle || 0)}°C
  2847. </p>
  2848. </div>
  2849. </NozzleSlotHoverCard>
  2850. ) : (
  2851. <>
  2852. <p className="text-[9px] text-bambu-gray">{t('printers.temperatures.nozzle')}</p>
  2853. <p className="text-[11px] text-white">
  2854. {Math.round(status.temperatures.nozzle || 0)}°C
  2855. </p>
  2856. </>
  2857. )}
  2858. </div>
  2859. <div className="text-center px-2 py-1.5 bg-bambu-dark rounded-lg flex-1 flex flex-col justify-center items-center">
  2860. <HeaterThermometer className="w-3.5 h-3.5 mb-0.5" color="text-blue-400" isHeating={bedHeating} />
  2861. <p className="text-[9px] text-bambu-gray">{t('printers.temperatures.bed')}</p>
  2862. <p className="text-[11px] text-white">
  2863. {Math.round(status.temperatures.bed || 0)}°C
  2864. </p>
  2865. </div>
  2866. {status.temperatures.chamber !== undefined && (
  2867. <div className="text-center px-2 py-1.5 bg-bambu-dark rounded-lg flex-1 flex flex-col justify-center items-center">
  2868. <HeaterThermometer className="w-3.5 h-3.5 mb-0.5" color="text-green-400" isHeating={chamberHeating} />
  2869. <p className="text-[9px] text-bambu-gray">{t('printers.temperatures.chamber')}</p>
  2870. <p className="text-[11px] text-white">
  2871. {Math.round(status.temperatures.chamber || 0)}°C
  2872. </p>
  2873. </div>
  2874. )}
  2875. {/* Active nozzle indicator for dual-nozzle printers */}
  2876. {isDualNozzle && (
  2877. <DualNozzleHoverCard
  2878. leftSlot={leftNozzleSlot}
  2879. rightSlot={rightNozzleSlot}
  2880. activeNozzle={activeNozzle}
  2881. filamentInfo={filamentInfo}
  2882. >
  2883. <div className="text-center px-3 py-1.5 bg-bambu-dark rounded-lg h-full flex flex-col justify-center items-center cursor-default" title={t('printers.activeNozzle', { nozzle: activeNozzle === 'L' ? t('common.left') : t('common.right') })}>
  2884. <NozzleIcon className="w-3.5 h-3.5 mb-0.5 text-amber-400" />
  2885. <div className="flex items-center gap-2">
  2886. <span className={`text-[11px] font-bold ${activeNozzle === 'L' ? 'text-amber-400' : 'text-gray-500'}`}>
  2887. L{leftNozzleSlot?.nozzle_diameter ? ` ${leftNozzleSlot.nozzle_diameter}` : ''}
  2888. </span>
  2889. <span className="text-[9px] text-bambu-gray/40">·</span>
  2890. <span className={`text-[11px] font-bold ${activeNozzle === 'R' ? 'text-amber-400' : 'text-gray-500'}`}>
  2891. R{rightNozzleSlot?.nozzle_diameter ? ` ${rightNozzleSlot.nozzle_diameter}` : ''}
  2892. </span>
  2893. </div>
  2894. <p className="text-[9px] text-bambu-gray">{t('printers.temperatures.nozzle')}</p>
  2895. </div>
  2896. </DualNozzleHoverCard>
  2897. )}
  2898. {/* H2C nozzle rack (tool-changer dock) — only show when rack nozzles exist (IDs >= 2) */}
  2899. {status.nozzle_rack && status.nozzle_rack.some(s => s.id >= 2) && (
  2900. <NozzleRackCard slots={status.nozzle_rack} filamentInfo={filamentInfo} />
  2901. )}
  2902. </div>
  2903. );
  2904. })()}
  2905. {viewMode === 'expanded' && showClearPlateButton && (
  2906. <button
  2907. type="button"
  2908. onClick={() => clearPlateMutation.mutate()}
  2909. disabled={clearPlateMutation.isPending || !hasPermission('printers:clear_plate')}
  2910. className="mt-2 w-full inline-flex items-center justify-center gap-2 px-3 py-1.5 rounded-lg bg-yellow-500/20 border border-yellow-400/40 text-yellow-400 hover:bg-yellow-500/30 transition-colors text-xs font-medium disabled:opacity-50"
  2911. title={!hasPermission('printers:clear_plate') ? t('printers.permission.noControl') : t('printers.plateStatus.markCleared')}
  2912. >
  2913. {clearPlateMutation.isPending ? (
  2914. <Loader2 className="w-3 h-3 animate-spin" />
  2915. ) : (
  2916. <PlateClearedIcon className="w-4 h-4" />
  2917. )}
  2918. {t('printers.plateStatus.markCleared')}
  2919. </button>
  2920. )}
  2921. {/* Controls - Fans + Print Buttons */}
  2922. {viewMode === 'expanded' && (() => {
  2923. // Determine print state for control buttons
  2924. const isRunning = status.state === 'RUNNING';
  2925. const isPaused = status.state === 'PAUSE';
  2926. const isPrinting = isRunning || isPaused;
  2927. const isControlBusy = stopPrintMutation.isPending || pausePrintMutation.isPending || resumePrintMutation.isPending;
  2928. // Fan data
  2929. const partFan = status.cooling_fan_speed;
  2930. const auxFan = status.big_fan1_speed;
  2931. const chamberFan = status.big_fan2_speed;
  2932. return (
  2933. <div className="mt-3">
  2934. {/* Section Header */}
  2935. <div className="flex items-center gap-2 mb-2">
  2936. <span className="text-[10px] uppercase tracking-wider text-bambu-gray font-medium">
  2937. {t('printers.controls')}
  2938. </span>
  2939. <div className="flex-1 h-px bg-bambu-dark-tertiary/30" />
  2940. </div>
  2941. <div className="flex flex-wrap items-start justify-between gap-x-2 gap-y-2">
  2942. {/* Left: Fan Status - always visible, dynamic coloring */}
  2943. <div className="flex flex-wrap items-center gap-x-2 gap-y-1.5 min-w-0">
  2944. {/* Part Cooling Fan */}
  2945. <div
  2946. className={`flex items-center gap-1 px-1.5 py-1 rounded ${partFan && partFan > 0 ? 'bg-cyan-500/10' : 'bg-bambu-dark'}`}
  2947. title={t('printers.fans.partCooling')}
  2948. >
  2949. <Fan className={`w-3.5 h-3.5 ${partFan && partFan > 0 ? 'text-cyan-400' : 'text-bambu-gray/50'}`} />
  2950. <span className={`text-[10px] ${partFan && partFan > 0 ? 'text-cyan-400' : 'text-bambu-gray/50'}`}>
  2951. {partFan ?? 0}%
  2952. </span>
  2953. </div>
  2954. {/* Auxiliary Fan */}
  2955. <div
  2956. className={`flex items-center gap-1 px-1.5 py-1 rounded ${auxFan && auxFan > 0 ? 'bg-blue-500/10' : 'bg-bambu-dark'}`}
  2957. title={t('printers.fans.auxiliary')}
  2958. >
  2959. <Wind className={`w-3.5 h-3.5 ${auxFan && auxFan > 0 ? 'text-blue-400' : 'text-bambu-gray/50'}`} />
  2960. <span className={`text-[10px] ${auxFan && auxFan > 0 ? 'text-blue-400' : 'text-bambu-gray/50'}`}>
  2961. {auxFan ?? 0}%
  2962. </span>
  2963. </div>
  2964. {/* Chamber Fan */}
  2965. <div
  2966. className={`flex items-center gap-1 px-1.5 py-1 rounded ${chamberFan && chamberFan > 0 ? 'bg-green-500/10' : 'bg-bambu-dark'}`}
  2967. title={t('printers.fans.chamber')}
  2968. >
  2969. <AirVent className={`w-3.5 h-3.5 ${chamberFan && chamberFan > 0 ? 'text-green-400' : 'text-bambu-gray/50'}`} />
  2970. <span className={`text-[10px] ${chamberFan && chamberFan > 0 ? 'text-green-400' : 'text-bambu-gray/50'}`}>
  2971. {chamberFan ?? 0}%
  2972. </span>
  2973. </div>
  2974. {/* Separator */}
  2975. <div className="w-px h-5 bg-bambu-gray/30" />
  2976. {/* Airduct Mode (P2S / X2D / H2*) */}
  2977. {(['P2S', 'X2D', 'H2D', 'H2C', 'H2S'].includes(printer.model ?? '')) && (() => {
  2978. const isHeating = status.airduct_mode === 1;
  2979. const Icon = isHeating ? Flame : Snowflake;
  2980. const color = isHeating ? 'text-orange-400' : 'text-sky-400';
  2981. const bg = isHeating ? 'bg-orange-500/10 hover:bg-orange-500/20' : 'bg-sky-500/10 hover:bg-sky-500/20';
  2982. return (
  2983. <div className="relative">
  2984. <button
  2985. onClick={() => setShowAirductMenu(showAirductMenu === printer.id ? null : printer.id)}
  2986. disabled={!hasPermission('printers:control')}
  2987. className={`flex items-center gap-1 px-1.5 py-1 rounded transition-colors ${bg} disabled:opacity-50 disabled:cursor-not-allowed`}
  2988. title={t('printers.airduct.title')}
  2989. >
  2990. <Icon className={`w-3.5 h-3.5 ${color}`} />
  2991. <span className={`text-[10px] ${color}`}>
  2992. {isHeating ? t('printers.airduct.heating') : t('printers.airduct.cooling')}
  2993. </span>
  2994. </button>
  2995. {showAirductMenu === printer.id && (
  2996. <>
  2997. <div className="fixed inset-0 z-40" onClick={() => setShowAirductMenu(null)} />
  2998. <div className="absolute bottom-full left-0 mb-1 z-50 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg shadow-lg py-1 min-w-[130px]">
  2999. {([
  3000. { mode: 'cooling', label: t('printers.airduct.cooling'), modeId: 0 },
  3001. { mode: 'heating', label: t('printers.airduct.heating'), modeId: 1 },
  3002. ] as const).map(({ mode, label, modeId }) => (
  3003. <button
  3004. key={mode}
  3005. onClick={() => {
  3006. airductMutation.mutate(mode);
  3007. setShowAirductMenu(null);
  3008. }}
  3009. className={`w-full text-left px-3 py-1.5 text-xs transition-colors flex items-center gap-2 ${
  3010. status.airduct_mode === modeId
  3011. ? 'text-bambu-green bg-bambu-green/10'
  3012. : 'text-white hover:bg-bambu-dark-tertiary'
  3013. }`}
  3014. >
  3015. {mode === 'heating' ? <Flame className="w-3 h-3" /> : <Snowflake className="w-3 h-3" />}
  3016. {label}
  3017. </button>
  3018. ))}
  3019. </div>
  3020. </>
  3021. )}
  3022. </div>
  3023. );
  3024. })()}
  3025. {/* Print Speed */}
  3026. {(() => {
  3027. const speedLabels: Record<number, string> = { 1: '50%', 2: '100%', 3: '124%', 4: '166%' };
  3028. const speedPct = speedLabels[status.speed_level] || '100%';
  3029. return (
  3030. <div className="relative">
  3031. <button
  3032. onClick={() => setShowSpeedMenu(showSpeedMenu === printer.id ? null : printer.id)}
  3033. disabled={!isPrinting || !hasPermission('printers:control')}
  3034. className={`flex items-center gap-1 px-1.5 py-1 rounded transition-colors ${
  3035. isPrinting
  3036. ? 'bg-amber-500/10 hover:bg-amber-500/20'
  3037. : 'bg-bambu-dark cursor-not-allowed'
  3038. }`}
  3039. title={isPrinting ? t('printers.speed.title') : undefined}
  3040. >
  3041. <Gauge className={`w-3.5 h-3.5 ${
  3042. isPrinting ? 'text-amber-400' : 'text-bambu-gray/50'
  3043. }`} />
  3044. <span className={`text-[10px] ${
  3045. isPrinting ? 'text-amber-400' : 'text-bambu-gray/50'
  3046. }`}>
  3047. {speedPct}
  3048. </span>
  3049. </button>
  3050. {showSpeedMenu === printer.id && (
  3051. <>
  3052. <div className="fixed inset-0 z-40" onClick={() => setShowSpeedMenu(null)} />
  3053. <div className="absolute bottom-full left-0 mb-1 z-50 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg shadow-lg py-1 min-w-[130px]">
  3054. {([
  3055. { mode: 1, label: t('printers.speed.silent') },
  3056. { mode: 2, label: t('printers.speed.standard') },
  3057. { mode: 3, label: t('printers.speed.sport') },
  3058. { mode: 4, label: t('printers.speed.ludicrous') },
  3059. ] as const).map(({ mode, label }) => (
  3060. <button
  3061. key={mode}
  3062. onClick={() => {
  3063. printSpeedMutation.mutate(mode);
  3064. setShowSpeedMenu(null);
  3065. }}
  3066. className={`w-full text-left px-3 py-1.5 text-xs transition-colors ${
  3067. status.speed_level === mode
  3068. ? 'text-bambu-green bg-bambu-green/10'
  3069. : 'text-white hover:bg-bambu-dark-tertiary'
  3070. }`}
  3071. >
  3072. {label}
  3073. </button>
  3074. ))}
  3075. </div>
  3076. </>
  3077. )}
  3078. </div>
  3079. );
  3080. })()}
  3081. {/* Separator */}
  3082. <div className="w-px h-5 bg-bambu-gray/30" />
  3083. {/* Bed Jog (Z-axis) — compact badge, popover holds the actual controls */}
  3084. {(() => {
  3085. const canControl = hasPermission('printers:control');
  3086. const disabled = isPrinting || !canControl;
  3087. const bambuIsPlateBelow = true; // positive Z moves plate away from nozzle
  3088. const requestJog = (direction: 1 | -1) => {
  3089. const signed = direction * bedJogStep * (bambuIsPlateBelow ? 1 : -1);
  3090. const warnedKey = `bambuddy.bedJog.warned.${printer.id}`;
  3091. const warned = (() => {
  3092. try { return sessionStorage.getItem(warnedKey) === '1'; }
  3093. catch { return false; }
  3094. })();
  3095. setShowBedJogMenu(null);
  3096. if (warned) {
  3097. bedJogMutation.mutate({ distance: signed, force: true });
  3098. } else {
  3099. setShowNotHomedModal({ distance: signed });
  3100. }
  3101. };
  3102. return (
  3103. <div className="relative">
  3104. <button
  3105. onClick={() => setShowBedJogMenu(showBedJogMenu === printer.id ? null : printer.id)}
  3106. disabled={disabled}
  3107. className={`flex items-center gap-1 px-1.5 py-1 rounded transition-colors ${
  3108. disabled
  3109. ? 'bg-bambu-dark cursor-not-allowed'
  3110. : 'bg-indigo-500/10 hover:bg-indigo-500/20'
  3111. }`}
  3112. title={!canControl ? t('printers.permission.noControl') : isPrinting ? t('printers.bedJog.disabledWhilePrinting') : t('printers.bedJog.title')}
  3113. >
  3114. <MoveVertical className={`w-3.5 h-3.5 ${disabled ? 'text-bambu-gray/50' : 'text-indigo-400'}`} />
  3115. <span className={`text-[10px] ${disabled ? 'text-bambu-gray/50' : 'text-indigo-400'}`}>
  3116. {t('printers.bedJog.bed')}
  3117. </span>
  3118. <span className={`text-[10px] tabular-nums opacity-70 ${disabled ? 'text-bambu-gray/50' : 'text-indigo-400'}`}>
  3119. {bedJogStep}mm
  3120. </span>
  3121. </button>
  3122. {showBedJogMenu === printer.id && (
  3123. <>
  3124. <div className="fixed inset-0 z-40" onClick={() => setShowBedJogMenu(null)} />
  3125. <div className="absolute bottom-full left-0 mb-1 z-50 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg shadow-lg p-2 min-w-[140px]">
  3126. <div className="flex items-center justify-between gap-1 mb-2">
  3127. <button
  3128. onClick={() => requestJog(-1)}
  3129. className="flex-1 flex items-center justify-center py-1.5 rounded bg-indigo-500/15 hover:bg-indigo-500/30 text-indigo-300"
  3130. aria-label={t('printers.bedJog.up')}
  3131. >
  3132. <ArrowUp className="w-4 h-4" />
  3133. </button>
  3134. <button
  3135. onClick={() => requestJog(1)}
  3136. className="flex-1 flex items-center justify-center py-1.5 rounded bg-indigo-500/15 hover:bg-indigo-500/30 text-indigo-300"
  3137. aria-label={t('printers.bedJog.down')}
  3138. >
  3139. <ArrowDown className="w-4 h-4" />
  3140. </button>
  3141. </div>
  3142. <div className="text-[9px] uppercase tracking-wider text-bambu-gray/70 px-1 mb-1">
  3143. {t('printers.bedJog.step')}
  3144. </div>
  3145. <div className="flex gap-1">
  3146. {[1, 10, 50].map((step) => (
  3147. <button
  3148. key={step}
  3149. onClick={() => setBedJogStep(step)}
  3150. className={`flex-1 px-1 py-1 rounded text-[10px] transition-colors ${
  3151. bedJogStep === step
  3152. ? 'bg-bambu-green/20 text-bambu-green'
  3153. : 'bg-bambu-dark text-bambu-gray hover:bg-bambu-dark-tertiary'
  3154. }`}
  3155. >
  3156. {step}
  3157. </button>
  3158. ))}
  3159. </div>
  3160. </div>
  3161. </>
  3162. )}
  3163. </div>
  3164. );
  3165. })()}
  3166. </div>
  3167. {/* Right: Print Control Buttons */}
  3168. <div className="flex items-center gap-2 flex-shrink-0 max-[550px]:self-start">
  3169. {/* Stop button */}
  3170. <button
  3171. onClick={() => setShowStopConfirm(true)}
  3172. disabled={!isPrinting || isControlBusy || !hasPermission('printers:control')}
  3173. className={`
  3174. flex items-center justify-center gap-1 px-3 py-1.5 rounded-lg text-xs font-medium
  3175. transition-colors
  3176. ${isPrinting && hasPermission('printers:control')
  3177. ? 'bg-red-500/20 text-red-400 hover:bg-red-500/30'
  3178. : 'bg-bambu-dark text-bambu-gray/50 cursor-not-allowed'
  3179. }
  3180. `}
  3181. title={!hasPermission('printers:control') ? t('printers.permission.noControl') : t('printers.stop')}
  3182. >
  3183. <Square className="w-3 h-3" />
  3184. {t('printers.stop')}
  3185. </button>
  3186. {/* Pause/Resume button */}
  3187. <button
  3188. onClick={() => isPaused ? setShowResumeConfirm(true) : setShowPauseConfirm(true)}
  3189. disabled={!isPrinting || isControlBusy || !hasPermission('printers:control')}
  3190. className={`
  3191. flex items-center justify-center gap-1 px-3 py-1.5 rounded-lg text-xs font-medium
  3192. transition-colors
  3193. ${isPrinting && hasPermission('printers:control')
  3194. ? isPaused
  3195. ? 'bg-bambu-green/20 text-bambu-green hover:bg-bambu-green/30'
  3196. : 'bg-yellow-500/20 text-yellow-400 hover:bg-yellow-500/30'
  3197. : 'bg-bambu-dark text-bambu-gray/50 cursor-not-allowed'
  3198. }
  3199. `}
  3200. title={!hasPermission('printers:control') ? t('printers.permission.noControl') : (isPaused ? t('printers.resume') : t('printers.pause'))}
  3201. >
  3202. {isPaused ? <Play className="w-3 h-3" /> : <Pause className="w-3 h-3" />}
  3203. {isPaused ? t('printers.resume') : t('printers.pause')}
  3204. </button>
  3205. </div>
  3206. </div>
  3207. </div>
  3208. );
  3209. })()}
  3210. {/* AMS Units - 2-Column Grid Layout */}
  3211. {(amsData?.length > 0 || status.vt_tray.length > 0) && viewMode === 'expanded' && (() => {
  3212. // Separate regular AMS (4-tray) from HT AMS (1-tray)
  3213. const regularAms = amsData.filter(ams => ams.tray.length > 1);
  3214. const htAms = amsData.filter(ams => ams.tray.length === 1);
  3215. const isDualNozzle = printer.nozzle_count === 2 || status?.temperatures?.nozzle_2 !== undefined;
  3216. return (
  3217. <div className="mt-3">
  3218. {/* Section Header */}
  3219. <div className="flex items-center gap-2 mb-2">
  3220. <span className="text-[10px] uppercase tracking-wider text-bambu-gray font-medium">
  3221. {t('printers.filaments')}
  3222. </span>
  3223. <div className="flex-1 h-px bg-bambu-dark-tertiary/30" />
  3224. </div>
  3225. {/* AMS Content */}
  3226. <div className="space-y-3">
  3227. {/* Row 1-2: Regular AMS (4-tray) in 2-column grid */}
  3228. {regularAms.length > 0 && (
  3229. <div className="grid grid-cols-2 gap-3">
  3230. {regularAms.map((ams) => {
  3231. const mappedExtruderId = amsExtruderMap[String(ams.id)];
  3232. const normalizedId = ams.id >= 128 ? ams.id - 128 : ams.id;
  3233. const extruderId = mappedExtruderId !== undefined ? mappedExtruderId : normalizedId;
  3234. const isLeftNozzle = extruderId === 1;
  3235. const isRightNozzle = extruderId === 0;
  3236. return (
  3237. <div key={ams.id} className="p-2.5 bg-bambu-dark rounded-lg border border-bambu-dark-tertiary/30">
  3238. {/* Header: Label + Stats (no icon) */}
  3239. <div className="flex items-center justify-between mb-2">
  3240. <div className="flex items-center gap-1.5">
  3241. {/* AMS name — hover to see serial, firmware, and edit friendly name */}
  3242. <AmsNameHoverCard
  3243. ams={ams}
  3244. printerId={printer.id}
  3245. label={getAmsLabel(ams.id, ams.tray.length)}
  3246. amsLabels={amsLabels}
  3247. canEdit={hasPermission('printers:update')}
  3248. onSaved={refetchAmsLabels}
  3249. >
  3250. <span className="text-[10px] text-white font-medium cursor-default select-none">
  3251. {amsLabels?.[ams.id] || getAmsLabel(ams.id, ams.tray.length)}
  3252. </span>
  3253. </AmsNameHoverCard>
  3254. {isDualNozzle && (isLeftNozzle || isRightNozzle) && (
  3255. <NozzleBadge side={isLeftNozzle ? 'L' : 'R'} />
  3256. )}
  3257. </div>
  3258. {(ams.humidity != null || ams.temp != null) && (
  3259. <div className="flex items-center gap-1.5 max-[550px]:flex-col max-[550px]:items-start">
  3260. {ams.humidity != null && (
  3261. <HumidityIndicator
  3262. humidity={ams.humidity}
  3263. goodThreshold={amsThresholds?.humidityGood}
  3264. fairThreshold={amsThresholds?.humidityFair}
  3265. onClick={() => setAmsHistoryModal({
  3266. amsId: ams.id,
  3267. amsLabel: getAmsLabel(ams.id, ams.tray.length),
  3268. mode: 'humidity',
  3269. })}
  3270. compact
  3271. />
  3272. )}
  3273. {ams.temp != null && (
  3274. <TemperatureIndicator
  3275. temp={ams.temp}
  3276. goodThreshold={amsThresholds?.tempGood}
  3277. fairThreshold={amsThresholds?.tempFair}
  3278. onClick={() => setAmsHistoryModal({
  3279. amsId: ams.id,
  3280. amsLabel: getAmsLabel(ams.id, ams.tray.length),
  3281. mode: 'temperature',
  3282. })}
  3283. compact
  3284. />
  3285. )}
  3286. {/* Drying button — only for AMS 2 Pro (n3f) and AMS-HT (n3s) */}
  3287. {status.supports_drying && (ams.module_type === 'n3f' || ams.module_type === 'n3s') && hasPermission('printers:control') && (
  3288. <button
  3289. disabled={!!(ams.dry_sf_reason?.length && ams.dry_time === 0)}
  3290. onClick={(e) => {
  3291. if (ams.dry_time > 0) {
  3292. stopDryingMutation.mutate(ams.id);
  3293. } else if (dryingPopoverAmsId === ams.id) {
  3294. setDryingPopoverAmsId(null);
  3295. } else {
  3296. const firstTray = ams.tray.find(t => t.tray_type);
  3297. const filType = (firstTray?.tray_type || 'PLA').split(' ')[0].toUpperCase();
  3298. const preset = dryingPresets[filType] || dryingPresets['PLA'];
  3299. const moduleType = ams.module_type as 'n3f' | 'n3s';
  3300. setDryingFilament(filType);
  3301. setDryingTemp(preset[moduleType] || preset.n3f);
  3302. setDryingDuration(moduleType === 'n3s' ? preset.n3s_hours : preset.n3f_hours);
  3303. setDryingRotateTray(false);
  3304. setDryingPopoverModuleType(ams.module_type);
  3305. setDryingPopoverAmsId(ams.id);
  3306. const rect = (e.currentTarget as HTMLElement).getBoundingClientRect();
  3307. setDryingPopoverPos(computePopoverPosition({ triggerRect: rect, popoverWidth: DRYING_POPOVER_WIDTH, estimatedHeight: DRYING_POPOVER_ESTIMATED_HEIGHT }));
  3308. }
  3309. }}
  3310. className={`flex items-center gap-0.5 px-1 py-0.5 rounded text-[9px] transition-colors ${
  3311. ams.dry_time > 0
  3312. ? 'bg-amber-500/20 text-amber-400'
  3313. : ams.dry_sf_reason?.length
  3314. ? 'bg-bambu-dark-tertiary/30 text-bambu-gray/50 cursor-not-allowed'
  3315. : 'bg-bambu-dark-tertiary/50 text-bambu-gray hover:text-white hover:bg-bambu-dark-tertiary'
  3316. }`}
  3317. title={ams.dry_time > 0 ? t('printers.drying.stop') : ams.dry_sf_reason?.length ? t('printers.drying.powerRequired') : t('printers.drying.start')}
  3318. >
  3319. <Flame className="w-3 h-3" />
  3320. </button>
  3321. )}
  3322. </div>
  3323. )}
  3324. </div>
  3325. {/* Drying status bar */}
  3326. {ams.dry_time > 0 && (
  3327. <div className="flex items-center gap-2 px-2 py-1 mb-1 bg-amber-500/10 border border-amber-500/20 rounded text-[9px]">
  3328. <Flame className="w-3 h-3 text-amber-400 shrink-0" />
  3329. <span className="text-amber-400 font-medium">{t('printers.drying.active')}</span>
  3330. <span className="text-amber-300/70">
  3331. {t('printers.drying.timeRemaining', {
  3332. time: ams.dry_time >= 60
  3333. ? `${Math.floor(ams.dry_time / 60)}h ${ams.dry_time % 60}m`
  3334. : `${ams.dry_time}m`
  3335. })}
  3336. </span>
  3337. <button
  3338. onClick={() => stopDryingMutation.mutate(ams.id)}
  3339. disabled={stopDryingMutation.isPending}
  3340. className="ml-auto text-amber-400 hover:text-amber-300 transition-colors disabled:opacity-50"
  3341. title={t('printers.drying.stop')}
  3342. >
  3343. <X className="w-3 h-3" />
  3344. </button>
  3345. </div>
  3346. )}
  3347. {/* Slots grid: 4 columns - always render 4 slots */}
  3348. <div className="grid grid-cols-4 gap-1.5">
  3349. {[0, 1, 2, 3].map((slotIdx) => {
  3350. // Find tray data for this slot (may be undefined if data incomplete)
  3351. // Use array index if available, as tray.id may not always be set
  3352. const tray = ams.tray[slotIdx] || ams.tray.find(t => t.id === slotIdx);
  3353. const hasFillLevel = tray?.tray_type && tray.remain >= 0;
  3354. const isEmpty = !tray?.tray_type;
  3355. const emptyKind = getEmptySlotKind(tray);
  3356. // Check if this is the currently loaded tray
  3357. // Global tray ID = ams.id * 4 + slot index (for standard AMS)
  3358. const globalTrayId = ams.id * 4 + slotIdx;
  3359. const isActive = effectiveTrayNow === globalTrayId;
  3360. // Get cloud preset info if available
  3361. const cloudInfo = tray?.tray_info_idx ? filamentInfo?.[tray.tray_info_idx] : null;
  3362. // Get saved slot preset mapping (for user-configured slots)
  3363. const slotPreset = slotPresets?.[globalTrayId];
  3364. // Fill level fallback chain: Spoolman → Inventory → AMS remain
  3365. const trayTag = (tray?.tray_uuid || tray?.tag_uid || getFallbackSpoolTag(printer.serial_number, ams.id, slotIdx))?.toUpperCase();
  3366. const linkedSpool = trayTag ? linkedSpools?.[trayTag] : undefined;
  3367. const spoolmanFill = getSpoolmanFillLevel(linkedSpool);
  3368. // Slot-assigned-only spool fill (no tag link required)
  3369. const slotAssignmentForFill = spoolmanEnabled && !spoolmanLoading
  3370. ? spoolmanSlotAssignments?.find(a => a.printer_id === printer.id && a.ams_id === ams.id && a.tray_id === slotIdx)
  3371. : undefined;
  3372. const slotSpoolForFill = slotAssignmentForFill
  3373. ? spoolmanSpools?.find(s => s.id === slotAssignmentForFill.spoolman_spool_id)
  3374. : undefined;
  3375. const slotSpoolFill = (slotSpoolForFill && (slotSpoolForFill.label_weight ?? 0) > 0)
  3376. ? Math.round(Math.max(0, (slotSpoolForFill.label_weight ?? 0) - slotSpoolForFill.weight_used) / (slotSpoolForFill.label_weight ?? 1) * 100)
  3377. : null;
  3378. const inventoryAssignment = onGetAssignment?.(printer.id, ams.id, slotIdx);
  3379. const inventoryFill = (() => {
  3380. const sp = inventoryAssignment?.spool;
  3381. if (sp && sp.label_weight > 0 && sp.weight_used != null) {
  3382. return Math.round(Math.max(0, sp.label_weight - sp.weight_used) / sp.label_weight * 100);
  3383. }
  3384. return null;
  3385. })();
  3386. // If inventory says 0% but AMS reports positive remain, prefer AMS
  3387. // (inventory weight_used may be stale or over-counted — #676)
  3388. const resolvedInventoryFill = (inventoryFill === 0 && hasFillLevel && tray.remain > 0)
  3389. ? null : inventoryFill;
  3390. const effectiveFill = spoolmanFill ?? slotSpoolFill ?? resolvedInventoryFill ?? (hasFillLevel ? tray.remain : null);
  3391. const fillSource = (spoolmanFill !== null || slotSpoolFill !== null) ? 'spoolman' as const
  3392. : resolvedInventoryFill !== null ? 'inventory' as const
  3393. : hasFillLevel ? 'ams' as const
  3394. : undefined;
  3395. // Build filament data for hover card
  3396. const filamentData = tray?.tray_type ? {
  3397. vendor: (isBambuLabSpool(tray) ? 'Bambu Lab' : 'Generic') as 'Bambu Lab' | 'Generic',
  3398. // Spoolman spool name wins over cloud lookup so a slot bound to
  3399. // a Spoolman spool shows that spool's preset name (e.g. "Devil
  3400. // Design PLA") instead of whatever the printer's filament_id
  3401. // resolves to in the cloud catalog (often "Generic PLA" for
  3402. // P-prefix local presets). Spoolman's filament.name is just the
  3403. // material+subtype ("PLA Basic"); prepend the spool's brand so
  3404. // the hover card shows "Devil Design PLA Basic" rather than the
  3405. // vendor-less form. Strip the "@<printer>..." suffix that
  3406. // BambuStudio appends to user-preset names.
  3407. profile: slotPreset?.preset_name || (slotSpoolForFill ? [slotSpoolForFill.brand, slotSpoolForFill.slicer_filament_name?.split('@')[0].trim() || slotSpoolForFill.material].filter(Boolean).join(' ').trim() : null) || inventoryAssignment?.spool?.slicer_filament_name || cloudInfo?.name || tray.tray_sub_brands || tray.tray_type,
  3408. colorName: getColorName(tray.tray_color || ''),
  3409. colorHex: tray.tray_color || null,
  3410. kFactor: formatKValue(tray.k),
  3411. fillLevel: effectiveFill,
  3412. trayUuid: tray.tray_uuid || null,
  3413. tagUid: tray.tag_uid || null,
  3414. fillSource,
  3415. } : null;
  3416. // Check if this specific slot is being refreshed
  3417. const isRefreshing = refreshingSlot?.amsId === ams.id &&
  3418. refreshingSlot?.slotId === slotIdx;
  3419. // Slot visual content (goes inside hover card)
  3420. const slotVisual = (
  3421. <div
  3422. className={`bg-bambu-dark-tertiary rounded p-1 text-center ${isEmpty ? 'opacity-50' : ''} ${isActive ? 'ring-2 ring-bambu-green ring-offset-1 ring-offset-bambu-dark' : ''}`}
  3423. >
  3424. {/* Filament color circle with 1-based slot number centered inside */}
  3425. <FilamentSlotCircle
  3426. trayColor={tray?.tray_color}
  3427. trayType={tray?.tray_type}
  3428. isEmpty={isEmpty}
  3429. emptyKind={emptyKind}
  3430. slotNumber={slotIdx + 1}
  3431. />
  3432. <div className="text-[9px] text-white font-bold truncate">
  3433. {tray?.tray_type || t('ams.slotEmpty')}
  3434. </div>
  3435. {/* Fill bar */}
  3436. <div className="mt-1 h-1.5 bg-black/30 rounded-full overflow-hidden">
  3437. {effectiveFill !== null && effectiveFill >= 0 && !isEmpty && tray && (
  3438. <div
  3439. className="h-full rounded-full transition-all"
  3440. style={{
  3441. width: `${effectiveFill}%`,
  3442. backgroundColor: getFillBarColor(effectiveFill),
  3443. }}
  3444. />
  3445. )}
  3446. </div>
  3447. </div>
  3448. );
  3449. // Wrapper with menu button, dropdown, and loading overlay (outside hover card)
  3450. return (
  3451. <div key={slotIdx} className="relative group">
  3452. {/* Loading overlay during RFID re-read */}
  3453. {isRefreshing && (
  3454. <div className="absolute inset-0 bg-bambu-dark-tertiary/80 rounded flex items-center justify-center z-20">
  3455. <RefreshCw className="w-4 h-4 text-bambu-green animate-spin" />
  3456. </div>
  3457. )}
  3458. {/* Menu button - appears on hover, hidden when printer busy */}
  3459. {status?.state !== 'RUNNING' && (
  3460. <button
  3461. onClick={(e) => {
  3462. e.stopPropagation();
  3463. setAmsSlotMenu(
  3464. amsSlotMenu?.amsId === ams.id && amsSlotMenu?.slotId === slotIdx
  3465. ? null
  3466. : { amsId: ams.id, slotId: slotIdx }
  3467. );
  3468. }}
  3469. className="absolute -top-1 -right-1 w-4 h-4 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-full flex items-center justify-center opacity-0 group-hover:opacity-100 transition-opacity z-10 hover:bg-bambu-dark-tertiary"
  3470. title={t('printers.slotOptions')}
  3471. >
  3472. <MoreVertical className="w-2.5 h-2.5 text-bambu-gray" />
  3473. </button>
  3474. )}
  3475. {/* Dropdown menu */}
  3476. {status?.state !== 'RUNNING' && amsSlotMenu?.amsId === ams.id && amsSlotMenu?.slotId === slotIdx && (
  3477. <div className="absolute top-full left-0 mt-1 z-50 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg shadow-xl py-1 min-w-[120px]">
  3478. <button
  3479. className={`w-full px-3 py-1.5 text-left text-xs flex items-center gap-2 ${
  3480. hasPermission('printers:ams_rfid')
  3481. ? 'text-white hover:bg-bambu-dark-tertiary'
  3482. : 'text-bambu-gray/50 cursor-not-allowed'
  3483. }`}
  3484. onClick={(e) => {
  3485. e.stopPropagation();
  3486. if (!hasPermission('printers:ams_rfid')) return;
  3487. refreshAmsSlotMutation.mutate({ amsId: ams.id, slotId: slotIdx });
  3488. setAmsSlotMenu(null);
  3489. }}
  3490. disabled={isRefreshing || !hasPermission('printers:ams_rfid')}
  3491. title={!hasPermission('printers:ams_rfid') ? t('printers.permission.noAmsRfid') : undefined}
  3492. >
  3493. <RefreshCw className={`w-3 h-3 ${isRefreshing ? 'animate-spin' : ''}`} />
  3494. {t('printers.rfid.reread')}
  3495. </button>
  3496. <button
  3497. className={`w-full px-3 py-1.5 text-left text-xs flex items-center gap-2 ${
  3498. hasPermission('printers:control')
  3499. ? 'text-white hover:bg-bambu-dark-tertiary'
  3500. : 'text-bambu-gray/50 cursor-not-allowed'
  3501. }`}
  3502. onClick={(e) => {
  3503. e.stopPropagation();
  3504. if (!hasPermission('printers:control')) return;
  3505. loadAmsTrayMutation.mutate({ trayId: ams.id * 4 + slotIdx });
  3506. setAmsSlotMenu(null);
  3507. }}
  3508. disabled={!hasPermission('printers:control')}
  3509. title={!hasPermission('printers:control') ? t('printers.permission.noControl') : undefined}
  3510. >
  3511. <LogIn className="w-3 h-3" />
  3512. {t('printers.ams.load')}
  3513. </button>
  3514. <button
  3515. className={`w-full px-3 py-1.5 text-left text-xs flex items-center gap-2 ${
  3516. hasPermission('printers:control')
  3517. ? 'text-white hover:bg-bambu-dark-tertiary'
  3518. : 'text-bambu-gray/50 cursor-not-allowed'
  3519. }`}
  3520. onClick={(e) => {
  3521. e.stopPropagation();
  3522. if (!hasPermission('printers:control')) return;
  3523. unloadAmsMutation.mutate();
  3524. setAmsSlotMenu(null);
  3525. }}
  3526. disabled={!hasPermission('printers:control')}
  3527. title={!hasPermission('printers:control') ? t('printers.permission.noControl') : undefined}
  3528. >
  3529. <LogOut className="w-3 h-3" />
  3530. {t('printers.ams.unload')}
  3531. </button>
  3532. </div>
  3533. )}
  3534. {/* Hover card wraps only the visual content */}
  3535. {filamentData ? (
  3536. <FilamentHoverCard
  3537. data={filamentData}
  3538. spoolman={{
  3539. enabled: spoolmanEnabled,
  3540. // #1457: slot assignment is the user's most explicit action — it must
  3541. // outrank the tag-link, which can be stale when a non-RFID slot's
  3542. // fallback tag is still attached to a previous spool in Spoolman.
  3543. linkedSpoolId: slotAssignmentForFill?.spoolman_spool_id
  3544. ?? (trayTag ? linkedSpools?.[trayTag]?.id : undefined),
  3545. spoolmanUrl,
  3546. syncMode: spoolmanSyncMode,
  3547. // Suppress Link button when slot is already occupied by ANY assignment
  3548. // (Spoolman SlotAssignment OR local SpoolAssignment). Phase 9 only
  3549. // suppressed for Spoolman; the maintainer screenshot shows the badge
  3550. // still appearing on slots with a local Devil Design PLA assigned.
  3551. onLinkSpool: (spoolmanEnabled && !slotAssignmentForFill && !inventoryAssignment) ? () => {
  3552. const linkTag = (filamentData.trayUuid || filamentData.tagUid || getFallbackSpoolTag(printer.serial_number, ams.id, slotIdx)).toUpperCase();
  3553. setLinkSpoolModal({
  3554. tagUid: filamentData.tagUid || linkTag,
  3555. trayUuid: filamentData.trayUuid || '',
  3556. printerId: printer.id,
  3557. amsId: ams.id,
  3558. trayId: slotIdx,
  3559. });
  3560. } : undefined,
  3561. onUnlinkSpool: linkedSpool?.id ? () => unlinkSpoolMutation.mutate(linkedSpool.id) : undefined,
  3562. }}
  3563. inventory={(() => {
  3564. if (spoolmanEnabled) {
  3565. if (spoolmanLoading) return undefined;
  3566. const slotAssignment = slotAssignmentForFill;
  3567. const spoolmanSpool = slotSpoolForFill;
  3568. return {
  3569. assignedSpool: spoolmanSpool ? {
  3570. id: spoolmanSpool.id,
  3571. material: spoolmanSpool.material,
  3572. brand: spoolmanSpool.brand ?? null,
  3573. color_name: spoolmanSpool.color_name ?? null,
  3574. remainingWeightGrams: spoolmanSpool.label_weight
  3575. ? Math.max(0, Math.round(spoolmanSpool.label_weight - spoolmanSpool.weight_used))
  3576. : undefined,
  3577. } : null,
  3578. onAssignSpool: () => setAssignSpoolModal({
  3579. printerId: printer.id,
  3580. amsId: ams.id,
  3581. trayId: slotIdx,
  3582. trayInfo: {
  3583. type: tray?.tray_type || filamentData.profile,
  3584. material: tray?.tray_type ?? undefined,
  3585. profile: filamentData.profile,
  3586. color: filamentData.colorHex || '',
  3587. location: `${getAmsLabel(ams.id, ams.tray.length)} Slot ${slotIdx + 1}`,
  3588. },
  3589. }),
  3590. onUnassignSpool: (spoolmanSpool && !isBambuLabSpool(tray)) ? () => onUnassignSpoolmanSpool?.(spoolmanSpool.id) : undefined,
  3591. isAssigned: !!slotAssignment || isBambuLabSpool(tray),
  3592. };
  3593. }
  3594. const assignment = onGetAssignment?.(printer.id, ams.id, slotIdx);
  3595. return {
  3596. assignedSpool: assignment?.spool ? {
  3597. id: assignment.spool.id,
  3598. material: assignment.spool.material,
  3599. brand: assignment.spool.brand,
  3600. color_name: assignment.spool.color_name,
  3601. remainingWeightGrams: Math.max(0, Math.round(assignment.spool.label_weight - assignment.spool.weight_used)),
  3602. } : null,
  3603. onAssignSpool: () => setAssignSpoolModal({
  3604. printerId: printer.id,
  3605. amsId: ams.id,
  3606. trayId: slotIdx,
  3607. trayInfo: {
  3608. type: tray?.tray_type || filamentData.profile,
  3609. material: tray?.tray_type ?? undefined,
  3610. profile: filamentData.profile,
  3611. color: filamentData.colorHex || '',
  3612. location: `${getAmsLabel(ams.id, ams.tray.length)} Slot ${slotIdx + 1}`,
  3613. },
  3614. }),
  3615. onUnassignSpool: (assignment && !isBambuLabSpool(tray)) ? () => onUnassignSpool?.(printer.id, ams.id, slotIdx) : undefined,
  3616. isAssigned: !!assignment || isBambuLabSpool(tray),
  3617. };
  3618. })()}
  3619. configureSlot={{
  3620. enabled: hasPermission('printers:control'),
  3621. onConfigure: () => setConfigureSlotModal({
  3622. amsId: ams.id,
  3623. trayId: slotIdx,
  3624. trayCount: ams.tray.length,
  3625. trayType: tray?.tray_type || undefined,
  3626. trayColor: tray?.tray_color || undefined,
  3627. traySubBrands: tray?.tray_sub_brands || undefined,
  3628. trayInfoIdx: tray?.tray_info_idx || undefined,
  3629. extruderId: mappedExtruderId,
  3630. caliIdx: tray?.cali_idx,
  3631. savedPresetId: slotPreset?.preset_id,
  3632. }),
  3633. }}
  3634. >
  3635. {slotVisual}
  3636. </FilamentHoverCard>
  3637. ) : (
  3638. <EmptySlotHoverCard
  3639. kind={emptyKind ?? undefined}
  3640. configureSlot={{
  3641. enabled: hasPermission('printers:control'),
  3642. onConfigure: () => setConfigureSlotModal({
  3643. amsId: ams.id,
  3644. trayId: slotIdx,
  3645. trayCount: ams.tray.length,
  3646. extruderId: mappedExtruderId,
  3647. }),
  3648. }}
  3649. onAssignSpool={() => setAssignSpoolModal({
  3650. printerId: printer.id,
  3651. amsId: ams.id,
  3652. trayId: slotIdx,
  3653. trayInfo: {
  3654. type: '',
  3655. material: undefined,
  3656. profile: '',
  3657. color: '',
  3658. location: `${getAmsLabel(ams.id, ams.tray.length)} Slot ${slotIdx + 1}`,
  3659. },
  3660. })}
  3661. >
  3662. {slotVisual}
  3663. </EmptySlotHoverCard>
  3664. )}
  3665. </div>
  3666. );
  3667. })}
  3668. </div>
  3669. </div>
  3670. );
  3671. })}
  3672. </div>
  3673. )}
  3674. {/* Row 3: HT AMS + External spools (same style as regular AMS, 4 across) */}
  3675. {(htAms.length > 0 || status.vt_tray.length > 0) && (
  3676. <div className="grid grid-cols-4 gap-3">
  3677. {/* HT AMS units - name/badge top, slot left, stats right */}
  3678. {htAms.map((ams) => {
  3679. const mappedExtruderId = amsExtruderMap[String(ams.id)];
  3680. const normalizedId = ams.id >= 128 ? ams.id - 128 : ams.id;
  3681. const extruderId = mappedExtruderId !== undefined ? mappedExtruderId : normalizedId;
  3682. const isLeftNozzle = extruderId === 1;
  3683. const isRightNozzle = extruderId === 0;
  3684. const tray = ams.tray[0];
  3685. const hasFillLevel = tray?.tray_type && tray.remain >= 0;
  3686. const isEmpty = !tray?.tray_type;
  3687. const emptyKind = getEmptySlotKind(tray);
  3688. // Check if this is the currently loaded tray
  3689. const globalTrayId = getGlobalTrayId(ams.id, tray?.id ?? 0, false);
  3690. const isActive = effectiveTrayNow === globalTrayId;
  3691. // Get cloud preset info if available
  3692. const cloudInfo = tray?.tray_info_idx ? filamentInfo?.[tray.tray_info_idx] : null;
  3693. // Get saved slot preset mapping (for user-configured slots)
  3694. const slotPreset = slotPresets?.[globalTrayId];
  3695. const htSlotId = tray?.id ?? 0;
  3696. // Fill level fallback chain: Spoolman → Inventory → AMS remain
  3697. const htTrayTag = (tray?.tray_uuid || tray?.tag_uid || getFallbackSpoolTag(printer.serial_number, ams.id, htSlotId))?.toUpperCase();
  3698. const htLinkedSpool = htTrayTag ? linkedSpools?.[htTrayTag] : undefined;
  3699. const htSpoolmanFill = getSpoolmanFillLevel(htLinkedSpool);
  3700. const htInventoryAssignment = onGetAssignment?.(printer.id, ams.id, htSlotId);
  3701. const htInventoryFill = (() => {
  3702. const sp = htInventoryAssignment?.spool;
  3703. if (sp && sp.label_weight > 0 && sp.weight_used != null) {
  3704. return Math.round(Math.max(0, sp.label_weight - sp.weight_used) / sp.label_weight * 100);
  3705. }
  3706. return null;
  3707. })();
  3708. // If inventory says 0% but AMS reports positive remain, prefer AMS (#676)
  3709. const htResolvedInventoryFill = (htInventoryFill === 0 && hasFillLevel && tray.remain > 0)
  3710. ? null : htInventoryFill;
  3711. // Slot-assigned-only fill (when spool has no NFC tag but is slot-assigned)
  3712. const htSlotAssignmentForFill = spoolmanEnabled && !spoolmanLoading
  3713. ? spoolmanSlotAssignments?.find(a => a.printer_id === printer.id && a.ams_id === ams.id && a.tray_id === htSlotId)
  3714. : undefined;
  3715. const htSlotSpoolForFill = htSlotAssignmentForFill
  3716. ? spoolmanSpools?.find(s => s.id === htSlotAssignmentForFill.spoolman_spool_id)
  3717. : undefined;
  3718. const htSlotSpoolFill = (htSlotSpoolForFill && (htSlotSpoolForFill.label_weight ?? 0) > 0)
  3719. ? Math.round(Math.max(0, (htSlotSpoolForFill.label_weight ?? 0) - htSlotSpoolForFill.weight_used) / (htSlotSpoolForFill.label_weight ?? 1) * 100)
  3720. : null;
  3721. const htEffectiveFill = htSpoolmanFill ?? htSlotSpoolFill ?? htResolvedInventoryFill ?? (hasFillLevel ? tray.remain : null);
  3722. const htFillSource = (htSpoolmanFill !== null || htSlotSpoolFill !== null) ? 'spoolman' as const
  3723. : htResolvedInventoryFill !== null ? 'inventory' as const
  3724. : hasFillLevel ? 'ams' as const
  3725. : undefined;
  3726. // Build filament data for hover card
  3727. const filamentData = tray?.tray_type ? {
  3728. vendor: (isBambuLabSpool(tray) ? 'Bambu Lab' : 'Generic') as 'Bambu Lab' | 'Generic',
  3729. profile: slotPreset?.preset_name || (htSlotSpoolForFill ? [htSlotSpoolForFill.brand, htSlotSpoolForFill.slicer_filament_name?.split('@')[0].trim() || htSlotSpoolForFill.material].filter(Boolean).join(' ').trim() : null) || htInventoryAssignment?.spool?.slicer_filament_name || cloudInfo?.name || tray.tray_sub_brands || tray.tray_type,
  3730. colorName: getColorName(tray.tray_color || ''),
  3731. colorHex: tray.tray_color || null,
  3732. kFactor: formatKValue(tray.k),
  3733. fillLevel: htEffectiveFill,
  3734. trayUuid: tray.tray_uuid || null,
  3735. tagUid: tray.tag_uid || null,
  3736. fillSource: htFillSource,
  3737. } : null;
  3738. // Check if this specific slot is being refreshed
  3739. const isHtRefreshing = refreshingSlot?.amsId === ams.id &&
  3740. refreshingSlot?.slotId === htSlotId;
  3741. // Slot visual content (goes inside hover card)
  3742. const slotVisual = (
  3743. <div
  3744. className={`bg-bambu-dark-tertiary rounded p-1 text-center ${isEmpty ? 'opacity-50' : ''} ${isActive ? 'ring-2 ring-bambu-green ring-offset-1 ring-offset-bambu-dark' : ''}`}
  3745. >
  3746. {/* Filament color circle with 1-based slot number centered inside */}
  3747. <FilamentSlotCircle
  3748. trayColor={tray?.tray_color}
  3749. trayType={tray?.tray_type}
  3750. isEmpty={isEmpty}
  3751. emptyKind={emptyKind}
  3752. slotNumber={1}
  3753. />
  3754. <div className="text-[9px] text-white font-bold truncate">
  3755. {tray?.tray_type || t('ams.slotEmpty')}
  3756. </div>
  3757. {/* Fill bar */}
  3758. <div className="mt-1 h-1.5 bg-black/30 rounded-full overflow-hidden">
  3759. {htEffectiveFill !== null && htEffectiveFill >= 0 && !isEmpty && (
  3760. <div
  3761. className="h-full rounded-full transition-all"
  3762. style={{
  3763. width: `${htEffectiveFill}%`,
  3764. backgroundColor: getFillBarColor(htEffectiveFill),
  3765. }}
  3766. />
  3767. )}
  3768. </div>
  3769. </div>
  3770. );
  3771. return (
  3772. <div key={ams.id} className="p-2.5 bg-bambu-dark rounded-lg border border-bambu-dark-tertiary/30">
  3773. {/* Row 1: Label + Nozzle + Drying */}
  3774. <div className="flex items-center gap-1 mb-2">
  3775. {/* AMS name — hover to see serial, firmware, and edit friendly name */}
  3776. <AmsNameHoverCard
  3777. ams={ams}
  3778. printerId={printer.id}
  3779. label={getAmsLabel(ams.id, ams.tray.length)}
  3780. amsLabels={amsLabels}
  3781. canEdit={hasPermission('printers:update')}
  3782. onSaved={refetchAmsLabels}
  3783. >
  3784. <span className="text-[10px] text-white font-medium cursor-default select-none">
  3785. {amsLabels?.[ams.id] || getAmsLabel(ams.id, ams.tray.length)}
  3786. </span>
  3787. </AmsNameHoverCard>
  3788. {isDualNozzle && (isLeftNozzle || isRightNozzle) && (
  3789. <NozzleBadge side={isLeftNozzle ? 'L' : 'R'} />
  3790. )}
  3791. {/* Drying button for HT AMS */}
  3792. {status.supports_drying && (ams.module_type === 'n3f' || ams.module_type === 'n3s') && hasPermission('printers:control') && (
  3793. <div className="relative ml-auto">
  3794. <button
  3795. onClick={(e) => {
  3796. if (ams.dry_time > 0) {
  3797. stopDryingMutation.mutate(ams.id);
  3798. } else if (dryingPopoverAmsId === ams.id) {
  3799. setDryingPopoverAmsId(null);
  3800. } else {
  3801. const firstTray = ams.tray.find(t => t.tray_type);
  3802. const filType = (firstTray?.tray_type || 'PLA').split(' ')[0].toUpperCase();
  3803. const preset = dryingPresets[filType] || dryingPresets['PLA'];
  3804. const moduleType = ams.module_type as 'n3f' | 'n3s';
  3805. setDryingFilament(filType);
  3806. setDryingTemp(preset[moduleType] || preset.n3f);
  3807. setDryingDuration(moduleType === 'n3s' ? preset.n3s_hours : preset.n3f_hours);
  3808. setDryingRotateTray(false);
  3809. setDryingPopoverModuleType(ams.module_type);
  3810. setDryingPopoverAmsId(ams.id);
  3811. const rect = (e.currentTarget as HTMLElement).getBoundingClientRect();
  3812. setDryingPopoverPos(computePopoverPosition({ triggerRect: rect, popoverWidth: DRYING_POPOVER_WIDTH, estimatedHeight: DRYING_POPOVER_ESTIMATED_HEIGHT }));
  3813. }
  3814. }}
  3815. className={`flex items-center gap-0.5 px-1 py-0.5 rounded text-[9px] transition-colors ${
  3816. ams.dry_time > 0
  3817. ? 'bg-amber-500/20 text-amber-400'
  3818. : 'bg-bambu-dark-tertiary/50 text-bambu-gray hover:text-white hover:bg-bambu-dark-tertiary'
  3819. }`}
  3820. title={ams.dry_time > 0 ? t('printers.drying.stop') : t('printers.drying.start')}
  3821. >
  3822. <Flame className="w-3 h-3" />
  3823. </button>
  3824. </div>
  3825. )}
  3826. </div>
  3827. {/* HT AMS drying status bar */}
  3828. {ams.dry_time > 0 && (
  3829. <div className="flex items-center gap-1.5 px-2 py-1 mb-1 bg-amber-500/10 border border-amber-500/20 rounded text-[9px] whitespace-nowrap overflow-hidden">
  3830. <Flame className="w-3 h-3 text-amber-400 shrink-0" />
  3831. <span className="text-amber-300/70 text-[8px] truncate">
  3832. {ams.dry_time >= 60
  3833. ? `${Math.floor(ams.dry_time / 60)}h ${ams.dry_time % 60}m`
  3834. : `${ams.dry_time}m`}
  3835. </span>
  3836. <button
  3837. onClick={() => stopDryingMutation.mutate(ams.id)}
  3838. disabled={stopDryingMutation.isPending}
  3839. className="ml-auto text-amber-400 hover:text-amber-300 transition-colors disabled:opacity-50 shrink-0"
  3840. title={t('printers.drying.stop')}
  3841. >
  3842. <X className="w-3 h-3" />
  3843. </button>
  3844. </div>
  3845. )}
  3846. {/* Row 2: Slot (left) + Stats (right stacked) */}
  3847. <div className="flex gap-1.5 max-[550px]:flex-col max-[550px]:items-start">
  3848. {/* Slot wrapper with menu button, dropdown, and loading overlay */}
  3849. <div className="relative group flex-1 max-[550px]:w-full">
  3850. {/* Loading overlay during RFID re-read */}
  3851. {isHtRefreshing && (
  3852. <div className="absolute inset-0 bg-bambu-dark-tertiary/80 rounded flex items-center justify-center z-20">
  3853. <RefreshCw className="w-4 h-4 text-bambu-green animate-spin" />
  3854. </div>
  3855. )}
  3856. {/* Menu button - appears on hover, hidden when printer busy */}
  3857. {status?.state !== 'RUNNING' && (
  3858. <button
  3859. onClick={(e) => {
  3860. e.stopPropagation();
  3861. setAmsSlotMenu(
  3862. amsSlotMenu?.amsId === ams.id && amsSlotMenu?.slotId === htSlotId
  3863. ? null
  3864. : { amsId: ams.id, slotId: htSlotId }
  3865. );
  3866. }}
  3867. className="absolute -top-1 -right-1 w-4 h-4 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-full flex items-center justify-center opacity-0 group-hover:opacity-100 transition-opacity z-10 hover:bg-bambu-dark-tertiary"
  3868. title={t('printers.slotOptions')}
  3869. >
  3870. <MoreVertical className="w-2.5 h-2.5 text-bambu-gray" />
  3871. </button>
  3872. )}
  3873. {/* Dropdown menu */}
  3874. {status?.state !== 'RUNNING' && amsSlotMenu?.amsId === ams.id && amsSlotMenu?.slotId === htSlotId && (
  3875. <div className="absolute top-full left-0 mt-1 z-50 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg shadow-xl py-1 min-w-[120px]">
  3876. <button
  3877. className={`w-full px-3 py-1.5 text-left text-xs flex items-center gap-2 ${
  3878. hasPermission('printers:ams_rfid')
  3879. ? 'text-white hover:bg-bambu-dark-tertiary'
  3880. : 'text-bambu-gray/50 cursor-not-allowed'
  3881. }`}
  3882. onClick={(e) => {
  3883. e.stopPropagation();
  3884. if (!hasPermission('printers:ams_rfid')) return;
  3885. refreshAmsSlotMutation.mutate({ amsId: ams.id, slotId: htSlotId });
  3886. setAmsSlotMenu(null);
  3887. }}
  3888. disabled={isHtRefreshing || !hasPermission('printers:ams_rfid')}
  3889. title={!hasPermission('printers:ams_rfid') ? t('printers.permission.noAmsRfid') : undefined}
  3890. >
  3891. <RefreshCw className={`w-3 h-3 ${isHtRefreshing ? 'animate-spin' : ''}`} />
  3892. {t('printers.rfid.reread')}
  3893. </button>
  3894. <button
  3895. className={`w-full px-3 py-1.5 text-left text-xs flex items-center gap-2 ${
  3896. hasPermission('printers:control')
  3897. ? 'text-white hover:bg-bambu-dark-tertiary'
  3898. : 'text-bambu-gray/50 cursor-not-allowed'
  3899. }`}
  3900. onClick={(e) => {
  3901. e.stopPropagation();
  3902. if (!hasPermission('printers:control')) return;
  3903. loadAmsTrayMutation.mutate({ trayId: ams.id * 4 + htSlotId });
  3904. setAmsSlotMenu(null);
  3905. }}
  3906. disabled={!hasPermission('printers:control')}
  3907. title={!hasPermission('printers:control') ? t('printers.permission.noControl') : undefined}
  3908. >
  3909. <LogIn className="w-3 h-3" />
  3910. {t('printers.ams.load')}
  3911. </button>
  3912. <button
  3913. className={`w-full px-3 py-1.5 text-left text-xs flex items-center gap-2 ${
  3914. hasPermission('printers:control')
  3915. ? 'text-white hover:bg-bambu-dark-tertiary'
  3916. : 'text-bambu-gray/50 cursor-not-allowed'
  3917. }`}
  3918. onClick={(e) => {
  3919. e.stopPropagation();
  3920. if (!hasPermission('printers:control')) return;
  3921. unloadAmsMutation.mutate();
  3922. setAmsSlotMenu(null);
  3923. }}
  3924. disabled={!hasPermission('printers:control')}
  3925. title={!hasPermission('printers:control') ? t('printers.permission.noControl') : undefined}
  3926. >
  3927. <LogOut className="w-3 h-3" />
  3928. {t('printers.ams.unload')}
  3929. </button>
  3930. </div>
  3931. )}
  3932. {/* Hover card wraps only the visual content */}
  3933. {filamentData ? (
  3934. <FilamentHoverCard
  3935. data={filamentData}
  3936. spoolman={{
  3937. enabled: spoolmanEnabled,
  3938. // #1457: slot assignment outranks tag-link (see top-level slot block).
  3939. linkedSpoolId: htSlotAssignmentForFill?.spoolman_spool_id
  3940. ?? (htTrayTag ? linkedSpools?.[htTrayTag]?.id : undefined),
  3941. spoolmanUrl,
  3942. syncMode: spoolmanSyncMode,
  3943. // Suppress Link button when slot is occupied by ANY assignment (Phase 13 P13-6d)
  3944. onLinkSpool: (spoolmanEnabled && !htSlotAssignmentForFill && !htInventoryAssignment) ? () => {
  3945. const linkTag = (filamentData.trayUuid || filamentData.tagUid || getFallbackSpoolTag(printer.serial_number, ams.id, htSlotId)).toUpperCase();
  3946. setLinkSpoolModal({
  3947. tagUid: filamentData.tagUid || linkTag,
  3948. trayUuid: filamentData.trayUuid || '',
  3949. printerId: printer.id,
  3950. amsId: ams.id,
  3951. trayId: htSlotId,
  3952. });
  3953. } : undefined,
  3954. onUnlinkSpool: htLinkedSpool?.id ? () => unlinkSpoolMutation.mutate(htLinkedSpool.id) : undefined,
  3955. }}
  3956. inventory={(() => {
  3957. if (spoolmanEnabled) {
  3958. if (spoolmanLoading) return undefined;
  3959. const slotAssignment = htSlotAssignmentForFill;
  3960. const spoolmanSpool = htSlotSpoolForFill;
  3961. return {
  3962. assignedSpool: spoolmanSpool ? {
  3963. id: spoolmanSpool.id,
  3964. material: spoolmanSpool.material,
  3965. brand: spoolmanSpool.brand ?? null,
  3966. color_name: spoolmanSpool.color_name ?? null,
  3967. remainingWeightGrams: spoolmanSpool.label_weight
  3968. ? Math.max(0, Math.round(spoolmanSpool.label_weight - spoolmanSpool.weight_used))
  3969. : undefined,
  3970. } : null,
  3971. onAssignSpool: () => setAssignSpoolModal({
  3972. printerId: printer.id,
  3973. amsId: ams.id,
  3974. trayId: htSlotId,
  3975. trayInfo: {
  3976. type: tray?.tray_type || filamentData.profile,
  3977. material: tray?.tray_type ?? undefined,
  3978. profile: filamentData.profile,
  3979. color: filamentData.colorHex || '',
  3980. location: getAmsLabel(ams.id, ams.tray.length),
  3981. },
  3982. }),
  3983. onUnassignSpool: (spoolmanSpool && !isBambuLabSpool(tray)) ? () => onUnassignSpoolmanSpool?.(spoolmanSpool.id) : undefined,
  3984. isAssigned: !!slotAssignment || isBambuLabSpool(tray),
  3985. };
  3986. }
  3987. const assignment = onGetAssignment?.(printer.id, ams.id, htSlotId);
  3988. return {
  3989. assignedSpool: assignment?.spool ? {
  3990. id: assignment.spool.id,
  3991. material: assignment.spool.material,
  3992. brand: assignment.spool.brand,
  3993. color_name: assignment.spool.color_name,
  3994. remainingWeightGrams: Math.max(0, Math.round(assignment.spool.label_weight - assignment.spool.weight_used)),
  3995. } : null,
  3996. onAssignSpool: () => setAssignSpoolModal({
  3997. printerId: printer.id,
  3998. amsId: ams.id,
  3999. trayId: htSlotId,
  4000. trayInfo: {
  4001. type: tray?.tray_type || filamentData.profile,
  4002. material: tray?.tray_type ?? undefined,
  4003. profile: filamentData.profile,
  4004. color: filamentData.colorHex || '',
  4005. location: getAmsLabel(ams.id, ams.tray.length),
  4006. },
  4007. }),
  4008. onUnassignSpool: (assignment && !isBambuLabSpool(tray)) ? () => onUnassignSpool?.(printer.id, ams.id, htSlotId) : undefined,
  4009. isAssigned: !!assignment || isBambuLabSpool(tray),
  4010. };
  4011. })()}
  4012. configureSlot={{
  4013. enabled: hasPermission('printers:control'),
  4014. onConfigure: () => setConfigureSlotModal({
  4015. amsId: ams.id,
  4016. trayId: htSlotId,
  4017. trayCount: ams.tray.length,
  4018. trayType: tray?.tray_type || undefined,
  4019. trayColor: tray?.tray_color || undefined,
  4020. traySubBrands: tray?.tray_sub_brands || undefined,
  4021. trayInfoIdx: tray?.tray_info_idx || undefined,
  4022. extruderId: mappedExtruderId,
  4023. caliIdx: tray?.cali_idx,
  4024. savedPresetId: slotPreset?.preset_id,
  4025. }),
  4026. }}
  4027. >
  4028. {slotVisual}
  4029. </FilamentHoverCard>
  4030. ) : (
  4031. <EmptySlotHoverCard
  4032. kind={emptyKind ?? undefined}
  4033. configureSlot={{
  4034. enabled: hasPermission('printers:control'),
  4035. onConfigure: () => setConfigureSlotModal({
  4036. amsId: ams.id,
  4037. trayId: htSlotId,
  4038. trayCount: ams.tray.length,
  4039. extruderId: mappedExtruderId,
  4040. }),
  4041. }}
  4042. onAssignSpool={() => setAssignSpoolModal({
  4043. printerId: printer.id,
  4044. amsId: ams.id,
  4045. trayId: htSlotId,
  4046. trayInfo: {
  4047. type: '',
  4048. material: undefined,
  4049. profile: '',
  4050. color: '',
  4051. location: getAmsLabel(ams.id, ams.tray.length),
  4052. },
  4053. })}
  4054. >
  4055. {slotVisual}
  4056. </EmptySlotHoverCard>
  4057. )}
  4058. </div>
  4059. {/* Stats stacked vertically: Temp on top, Humidity below */}
  4060. {(ams.humidity != null || ams.temp != null) && (
  4061. <div className="flex flex-col justify-center gap-1 shrink-0 max-[550px]:w-full">
  4062. {ams.temp != null && (
  4063. <TemperatureIndicator
  4064. temp={ams.temp}
  4065. goodThreshold={amsThresholds?.tempGood}
  4066. fairThreshold={amsThresholds?.tempFair}
  4067. onClick={() => setAmsHistoryModal({
  4068. amsId: ams.id,
  4069. amsLabel: getAmsLabel(ams.id, ams.tray.length),
  4070. mode: 'temperature',
  4071. })}
  4072. compact
  4073. />
  4074. )}
  4075. {ams.humidity != null && (
  4076. <HumidityIndicator
  4077. humidity={ams.humidity}
  4078. goodThreshold={amsThresholds?.humidityGood}
  4079. fairThreshold={amsThresholds?.humidityFair}
  4080. onClick={() => setAmsHistoryModal({
  4081. amsId: ams.id,
  4082. amsLabel: getAmsLabel(ams.id, ams.tray.length),
  4083. mode: 'humidity',
  4084. })}
  4085. compact
  4086. />
  4087. )}
  4088. </div>
  4089. )}
  4090. </div>
  4091. </div>
  4092. );
  4093. })}
  4094. {/* External spool(s) - grouped in one card like regular AMS */}
  4095. {status.vt_tray.length > 0 && (
  4096. <div className={`p-2.5 bg-bambu-dark rounded-lg border border-bambu-dark-tertiary/30 ${status.vt_tray.length === 1 ? 'max-w-[50%]' : ''}`}>
  4097. <div className="flex items-center gap-1 mb-2">
  4098. <span className="text-[10px] text-white font-medium">{t('printers.external')}</span>
  4099. </div>
  4100. <div className={`grid ${status.vt_tray.length > 1 ? 'grid-cols-2' : 'grid-cols-1'} gap-1.5`}>
  4101. {[...status.vt_tray].sort((a, b) => (a.id ?? 254) - (b.id ?? 254)).map((extTray) => {
  4102. const extTrayId = extTray.id ?? 254;
  4103. // On dual-nozzle (H2C/H2D), tray_now=254 means "external spool"
  4104. // generically — use active_extruder to determine L vs R:
  4105. // extruder 1=left → Ext-L (id=254), extruder 0=right → Ext-R (id=255)
  4106. const isExtActive = isDualNozzle && effectiveTrayNow === 254
  4107. ? (extTrayId === 254 && status.active_extruder === 1) ||
  4108. (extTrayId === 255 && status.active_extruder === 0)
  4109. : effectiveTrayNow === extTrayId;
  4110. const slotTrayId = extTrayId - 254; // 0 or 1
  4111. const extLabel = isDualNozzle
  4112. ? (extTrayId === 254 ? t('printers.extL') : t('printers.extR'))
  4113. : '';
  4114. const extCloudInfo = extTray.tray_info_idx ? filamentInfo?.[extTray.tray_info_idx] : null;
  4115. const extSlotPreset = slotPresets?.[255 * 4 + slotTrayId];
  4116. const extTrayTag = (extTray.tray_uuid || extTray.tag_uid || getFallbackSpoolTag(printer.serial_number, 255, slotTrayId))?.toUpperCase();
  4117. const extLinkedSpool = extTrayTag ? linkedSpools?.[extTrayTag] : undefined;
  4118. const extSpoolmanFill = getSpoolmanFillLevel(extLinkedSpool);
  4119. const extInventoryAssignment = onGetAssignment?.(printer.id, 255, slotTrayId);
  4120. const extInventoryFill = (() => {
  4121. const sp = extInventoryAssignment?.spool;
  4122. if (sp && sp.label_weight > 0 && sp.weight_used != null) {
  4123. return Math.round(Math.max(0, sp.label_weight - sp.weight_used) / sp.label_weight * 100);
  4124. }
  4125. return null;
  4126. })();
  4127. const extHasFillLevel = extTray.tray_type && extTray.remain >= 0;
  4128. // If inventory says 0% but AMS reports positive remain, prefer AMS (#676)
  4129. const extResolvedInventoryFill = (extInventoryFill === 0 && extHasFillLevel && extTray.remain > 0)
  4130. ? null : extInventoryFill;
  4131. // Slot-assigned-only fill (when spool has no NFC tag but is slot-assigned)
  4132. const extSlotAssignmentForFill = spoolmanEnabled && !spoolmanLoading
  4133. ? spoolmanSlotAssignments?.find(a => a.printer_id === printer.id && a.ams_id === 255 && a.tray_id === slotTrayId)
  4134. : undefined;
  4135. const extSlotSpoolForFill = extSlotAssignmentForFill
  4136. ? spoolmanSpools?.find(s => s.id === extSlotAssignmentForFill.spoolman_spool_id)
  4137. : undefined;
  4138. const extSlotSpoolFill = (extSlotSpoolForFill && (extSlotSpoolForFill.label_weight ?? 0) > 0)
  4139. ? Math.round(Math.max(0, (extSlotSpoolForFill.label_weight ?? 0) - extSlotSpoolForFill.weight_used) / (extSlotSpoolForFill.label_weight ?? 1) * 100)
  4140. : null;
  4141. const extEffectiveFill = extSpoolmanFill ?? extSlotSpoolFill ?? extResolvedInventoryFill ?? (extHasFillLevel ? extTray.remain : null);
  4142. const extFillSource = (extSpoolmanFill !== null || extSlotSpoolFill !== null) ? 'spoolman' as const
  4143. : extResolvedInventoryFill !== null ? 'inventory' as const
  4144. : extHasFillLevel ? 'ams' as const
  4145. : undefined;
  4146. const extFilamentData = {
  4147. vendor: (isBambuLabSpool(extTray) ? 'Bambu Lab' : 'Generic') as 'Bambu Lab' | 'Generic',
  4148. profile: extSlotPreset?.preset_name || (extSlotSpoolForFill ? [extSlotSpoolForFill.brand, extSlotSpoolForFill.slicer_filament_name?.split('@')[0].trim() || extSlotSpoolForFill.material].filter(Boolean).join(' ').trim() : null) || extInventoryAssignment?.spool?.slicer_filament_name || extCloudInfo?.name || extTray.tray_sub_brands || extTray.tray_type || 'Unknown',
  4149. colorName: getColorName(extTray.tray_color || ''),
  4150. colorHex: extTray.tray_color || null,
  4151. kFactor: formatKValue(extTray.k),
  4152. fillLevel: extEffectiveFill,
  4153. trayUuid: extTray.tray_uuid || null,
  4154. tagUid: extTray.tag_uid || null,
  4155. fillSource: extFillSource,
  4156. };
  4157. const isEmpty = !extTray.tray_type;
  4158. const emptyKind = getEmptySlotKind(extTray);
  4159. const extSlotContent = (
  4160. <div className={`bg-bambu-dark-tertiary rounded p-1 text-center ${isEmpty ? 'opacity-50' : ''} ${isExtActive ? 'ring-2 ring-bambu-green ring-offset-1 ring-offset-bambu-dark' : ''}`}>
  4161. {/* Filament color circle with 1-based slot number centered inside */}
  4162. <FilamentSlotCircle
  4163. trayColor={extTray.tray_color}
  4164. trayType={extTray.tray_type}
  4165. isEmpty={isEmpty}
  4166. emptyKind={emptyKind}
  4167. slotNumber={slotTrayId + 1}
  4168. />
  4169. <div className={`text-[9px] font-bold truncate ${isEmpty ? 'text-white/40' : 'text-white'}`}>
  4170. {extTray.tray_type || t('ams.slotEmpty')}
  4171. </div>
  4172. <div className="mt-1 h-1.5 bg-black/30 rounded-full overflow-hidden">
  4173. {extEffectiveFill !== null && extEffectiveFill >= 0 && !isEmpty && (
  4174. <div
  4175. className="h-full rounded-full transition-all"
  4176. style={{
  4177. width: `${extEffectiveFill}%`,
  4178. backgroundColor: getFillBarColor(extEffectiveFill),
  4179. }}
  4180. />
  4181. )}
  4182. </div>
  4183. {extLabel && <div className="text-[7px] text-white/40 mt-0.5 truncate">{extLabel}</div>}
  4184. </div>
  4185. );
  4186. const extMenuKey = 255 * 10 + slotTrayId; // unique slotId space for external menu state
  4187. const isExtMenuOpen = amsSlotMenu?.amsId === 255 && amsSlotMenu?.slotId === extMenuKey;
  4188. return (
  4189. <div key={extTrayId} className="relative group">
  4190. {/* Menu button - appears on hover, hidden when printer busy */}
  4191. {status?.state !== 'RUNNING' && (
  4192. <button
  4193. onClick={(e) => {
  4194. e.stopPropagation();
  4195. setAmsSlotMenu(isExtMenuOpen ? null : { amsId: 255, slotId: extMenuKey });
  4196. }}
  4197. className="absolute -top-1 -right-1 w-4 h-4 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-full flex items-center justify-center opacity-0 group-hover:opacity-100 transition-opacity z-10 hover:bg-bambu-dark-tertiary"
  4198. title={t('printers.slotOptions')}
  4199. >
  4200. <MoreVertical className="w-2.5 h-2.5 text-bambu-gray" />
  4201. </button>
  4202. )}
  4203. {/* Dropdown menu */}
  4204. {status?.state !== 'RUNNING' && isExtMenuOpen && (
  4205. <div className="absolute top-full left-0 mt-1 z-50 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg shadow-xl py-1 min-w-[120px]">
  4206. <button
  4207. className={`w-full px-3 py-1.5 text-left text-xs flex items-center gap-2 ${
  4208. hasPermission('printers:control')
  4209. ? 'text-white hover:bg-bambu-dark-tertiary'
  4210. : 'text-bambu-gray/50 cursor-not-allowed'
  4211. }`}
  4212. onClick={(e) => {
  4213. e.stopPropagation();
  4214. if (!hasPermission('printers:control')) return;
  4215. loadAmsTrayMutation.mutate({ trayId: extTrayId });
  4216. setAmsSlotMenu(null);
  4217. }}
  4218. disabled={!hasPermission('printers:control')}
  4219. title={!hasPermission('printers:control') ? t('printers.permission.noControl') : undefined}
  4220. >
  4221. <LogIn className="w-3 h-3" />
  4222. {t('printers.ams.load')}
  4223. </button>
  4224. <button
  4225. className={`w-full px-3 py-1.5 text-left text-xs flex items-center gap-2 ${
  4226. hasPermission('printers:control')
  4227. ? 'text-white hover:bg-bambu-dark-tertiary'
  4228. : 'text-bambu-gray/50 cursor-not-allowed'
  4229. }`}
  4230. onClick={(e) => {
  4231. e.stopPropagation();
  4232. if (!hasPermission('printers:control')) return;
  4233. unloadAmsMutation.mutate();
  4234. setAmsSlotMenu(null);
  4235. }}
  4236. disabled={!hasPermission('printers:control')}
  4237. title={!hasPermission('printers:control') ? t('printers.permission.noControl') : undefined}
  4238. >
  4239. <LogOut className="w-3 h-3" />
  4240. {t('printers.ams.unload')}
  4241. </button>
  4242. </div>
  4243. )}
  4244. {!isEmpty ? (
  4245. <FilamentHoverCard
  4246. data={extFilamentData}
  4247. spoolman={{
  4248. enabled: spoolmanEnabled,
  4249. // #1457: slot assignment outranks tag-link (see top-level slot block).
  4250. linkedSpoolId: extSlotAssignmentForFill?.spoolman_spool_id
  4251. ?? (extTrayTag ? linkedSpools?.[extTrayTag]?.id : undefined),
  4252. spoolmanUrl,
  4253. syncMode: spoolmanSyncMode,
  4254. // Suppress Link button when slot is occupied by ANY assignment (Phase 13 P13-6d)
  4255. onLinkSpool: (spoolmanEnabled && !extSlotAssignmentForFill && !extInventoryAssignment) ? () => {
  4256. const linkTag = (extFilamentData.trayUuid || extFilamentData.tagUid || getFallbackSpoolTag(printer.serial_number, 255, slotTrayId)).toUpperCase();
  4257. setLinkSpoolModal({
  4258. tagUid: extFilamentData.tagUid || linkTag,
  4259. trayUuid: extFilamentData.trayUuid || '',
  4260. printerId: printer.id,
  4261. amsId: 255,
  4262. trayId: slotTrayId,
  4263. });
  4264. } : undefined,
  4265. onUnlinkSpool: extLinkedSpool?.id ? () => unlinkSpoolMutation.mutate(extLinkedSpool.id) : undefined,
  4266. }}
  4267. inventory={(() => {
  4268. if (spoolmanEnabled) {
  4269. if (spoolmanLoading) return undefined;
  4270. const slotAssignment = extSlotAssignmentForFill;
  4271. const spoolmanSpool = extSlotSpoolForFill;
  4272. return {
  4273. assignedSpool: spoolmanSpool ? {
  4274. id: spoolmanSpool.id,
  4275. material: spoolmanSpool.material,
  4276. brand: spoolmanSpool.brand ?? null,
  4277. color_name: spoolmanSpool.color_name ?? null,
  4278. remainingWeightGrams: spoolmanSpool.label_weight
  4279. ? Math.max(0, Math.round(spoolmanSpool.label_weight - spoolmanSpool.weight_used))
  4280. : undefined,
  4281. } : null,
  4282. onAssignSpool: () => setAssignSpoolModal({
  4283. printerId: printer.id,
  4284. amsId: 255,
  4285. trayId: slotTrayId,
  4286. trayInfo: {
  4287. type: extTray.tray_type || extFilamentData.profile,
  4288. material: extTray.tray_type ?? undefined,
  4289. profile: extFilamentData.profile,
  4290. color: extFilamentData.colorHex || '',
  4291. location: extLabel || t('printers.external'),
  4292. },
  4293. }),
  4294. onUnassignSpool: (spoolmanSpool && !isBambuLabSpool(extTray)) ? () => onUnassignSpoolmanSpool?.(spoolmanSpool.id) : undefined,
  4295. isAssigned: !!slotAssignment || isBambuLabSpool(extTray),
  4296. };
  4297. }
  4298. const assignment = onGetAssignment?.(printer.id, 255, slotTrayId);
  4299. return {
  4300. assignedSpool: assignment?.spool ? {
  4301. id: assignment.spool.id,
  4302. material: assignment.spool.material,
  4303. brand: assignment.spool.brand,
  4304. color_name: assignment.spool.color_name,
  4305. remainingWeightGrams: Math.max(0, Math.round(assignment.spool.label_weight - assignment.spool.weight_used)),
  4306. } : null,
  4307. onAssignSpool: () => setAssignSpoolModal({
  4308. printerId: printer.id,
  4309. amsId: 255,
  4310. trayId: slotTrayId,
  4311. trayInfo: {
  4312. type: extTray.tray_type || extFilamentData.profile,
  4313. material: extTray.tray_type ?? undefined,
  4314. profile: extFilamentData.profile,
  4315. color: extFilamentData.colorHex || '',
  4316. location: extLabel || t('printers.external'),
  4317. },
  4318. }),
  4319. onUnassignSpool: (assignment && !isBambuLabSpool(extTray)) ? () => onUnassignSpool?.(printer.id, 255, slotTrayId) : undefined,
  4320. isAssigned: !!assignment || isBambuLabSpool(extTray),
  4321. };
  4322. })()}
  4323. configureSlot={{
  4324. enabled: hasPermission('printers:control'),
  4325. onConfigure: () => setConfigureSlotModal({
  4326. amsId: 255,
  4327. trayId: slotTrayId,
  4328. trayCount: 1,
  4329. trayType: extTray.tray_type || undefined,
  4330. trayColor: extTray.tray_color || undefined,
  4331. traySubBrands: extTray.tray_sub_brands || undefined,
  4332. trayInfoIdx: extTray.tray_info_idx || undefined,
  4333. extruderId: isDualNozzle ? (extTrayId === 254 ? 1 : 0) : undefined,
  4334. caliIdx: extTray.cali_idx,
  4335. savedPresetId: extSlotPreset?.preset_id,
  4336. }),
  4337. }}
  4338. >
  4339. {extSlotContent}
  4340. </FilamentHoverCard>
  4341. ) : (
  4342. <EmptySlotHoverCard
  4343. kind={emptyKind ?? undefined}
  4344. configureSlot={{
  4345. enabled: hasPermission('printers:control'),
  4346. onConfigure: () => setConfigureSlotModal({
  4347. amsId: 255,
  4348. trayId: slotTrayId,
  4349. trayCount: 1,
  4350. extruderId: isDualNozzle ? (extTrayId === 254 ? 1 : 0) : undefined,
  4351. }),
  4352. }}
  4353. onAssignSpool={() => setAssignSpoolModal({
  4354. printerId: printer.id,
  4355. amsId: 255,
  4356. trayId: slotTrayId,
  4357. trayInfo: {
  4358. type: '',
  4359. material: undefined,
  4360. profile: '',
  4361. color: '',
  4362. location: `External Slot ${slotTrayId + 1}`,
  4363. },
  4364. })}
  4365. >
  4366. {extSlotContent}
  4367. </EmptySlotHoverCard>
  4368. )}
  4369. </div>
  4370. );
  4371. })}
  4372. </div>
  4373. </div>
  4374. )}
  4375. </div>
  4376. )}
  4377. </div>
  4378. </div>
  4379. );
  4380. })()}
  4381. </>
  4382. )}
  4383. {/* Smart Plug Controls - hidden in compact mode */}
  4384. {smartPlug && viewMode === 'expanded' && (
  4385. <div className="mt-4 pt-4 border-t border-bambu-dark-tertiary">
  4386. <div className="flex items-center gap-3">
  4387. {/* Plug name and status */}
  4388. <div className="flex items-center gap-2 min-w-0">
  4389. <Zap className="w-4 h-4 text-bambu-gray flex-shrink-0" />
  4390. <span className="text-sm text-white truncate">{smartPlug.name}</span>
  4391. {plugStatus && (
  4392. <span
  4393. className={`text-xs px-1.5 py-0.5 rounded flex-shrink-0 ${
  4394. plugStatus.state === 'ON'
  4395. ? 'bg-bambu-green/20 text-bambu-green'
  4396. : plugStatus.state === 'OFF'
  4397. ? 'bg-red-500/20 text-red-400'
  4398. : 'bg-bambu-gray/20 text-bambu-gray'
  4399. }`}
  4400. >
  4401. {plugStatus.state || '?'}
  4402. {plugStatus.state === 'ON' && plugStatus.energy?.power != null && (
  4403. <span className="text-yellow-400 ml-1.5">· {Math.round(plugStatus.energy.power)}W</span>
  4404. )}
  4405. </span>
  4406. )}
  4407. </div>
  4408. {/* Spacer */}
  4409. <div className="flex-1" />
  4410. {/* Power buttons */}
  4411. <div className="flex items-center gap-1">
  4412. <button
  4413. onClick={() => setShowPowerOnConfirm(true)}
  4414. disabled={powerControlMutation.isPending || plugStatus?.state === 'ON' || !hasPermission('smart_plugs:control')}
  4415. className={`px-2 py-1 text-xs rounded transition-colors flex items-center gap-1 ${
  4416. !hasPermission('smart_plugs:control')
  4417. ? 'bg-bambu-dark text-bambu-gray/50 cursor-not-allowed'
  4418. : plugStatus?.state === 'ON'
  4419. ? 'bg-bambu-green text-white'
  4420. : 'bg-bambu-dark text-bambu-gray hover:text-white hover:bg-bambu-dark-tertiary'
  4421. }`}
  4422. title={!hasPermission('smart_plugs:control') ? t('printers.permission.noSmartPlugControl') : undefined}
  4423. >
  4424. <Power className="w-3 h-3" />
  4425. On
  4426. </button>
  4427. <button
  4428. onClick={() => setShowPowerOffConfirm(true)}
  4429. disabled={powerControlMutation.isPending || plugStatus?.state === 'OFF' || !hasPermission('smart_plugs:control')}
  4430. className={`px-2 py-1 text-xs rounded transition-colors flex items-center gap-1 ${
  4431. !hasPermission('smart_plugs:control')
  4432. ? 'bg-bambu-dark text-bambu-gray/50 cursor-not-allowed'
  4433. : plugStatus?.state === 'OFF'
  4434. ? 'bg-red-500/30 text-red-400'
  4435. : 'bg-bambu-dark text-bambu-gray hover:text-white hover:bg-bambu-dark-tertiary'
  4436. }`}
  4437. title={!hasPermission('smart_plugs:control') ? t('printers.permission.noSmartPlugControl') : undefined}
  4438. >
  4439. <PowerOff className="w-3 h-3" />
  4440. Off
  4441. </button>
  4442. </div>
  4443. {/* Auto-off toggle */}
  4444. <div className="flex items-center gap-2 flex-shrink-0">
  4445. <span className={`text-xs hidden sm:inline ${smartPlug.auto_off_executed ? 'text-bambu-green' : 'text-bambu-gray'}`}>
  4446. {smartPlug.auto_off_executed ? 'Auto-off done' : 'Auto-off'}
  4447. </span>
  4448. <button
  4449. onClick={() => toggleAutoOffMutation.mutate(!smartPlug.auto_off)}
  4450. disabled={toggleAutoOffMutation.isPending || smartPlug.auto_off_executed || !hasPermission('smart_plugs:control')}
  4451. title={!hasPermission('smart_plugs:control') ? t('printers.permission.noSmartPlugControl') : (smartPlug.auto_off_executed ? t('printers.autoOffExecuted') : t('printers.autoOffAfterPrint'))}
  4452. className={`relative w-9 h-5 rounded-full transition-colors flex-shrink-0 ${
  4453. !hasPermission('smart_plugs:control')
  4454. ? 'bg-bambu-dark-tertiary/50 cursor-not-allowed'
  4455. : smartPlug.auto_off_executed
  4456. ? 'bg-bambu-green/50 cursor-not-allowed'
  4457. : smartPlug.auto_off ? 'bg-bambu-green' : 'bg-bambu-dark-tertiary'
  4458. }`}
  4459. >
  4460. <span
  4461. className={`absolute top-[2px] left-[2px] w-4 h-4 bg-white rounded-full transition-transform ${
  4462. smartPlug.auto_off || smartPlug.auto_off_executed ? 'translate-x-4' : 'translate-x-0'
  4463. }`}
  4464. />
  4465. </button>
  4466. </div>
  4467. </div>
  4468. {/* HA entity buttons row */}
  4469. {scriptPlugs && scriptPlugs.length > 0 && (
  4470. <div className="flex items-center gap-2 mt-2 pt-2 border-t border-bambu-dark-tertiary/50">
  4471. <Home className="w-3.5 h-3.5 text-blue-400 flex-shrink-0" />
  4472. <span className="text-xs text-bambu-gray">HA:</span>
  4473. <div className="flex flex-wrap gap-1">
  4474. {scriptPlugs.map(script => {
  4475. const isScript = script.ha_entity_id?.startsWith('script.');
  4476. return (
  4477. <button
  4478. key={script.id}
  4479. onClick={() => {
  4480. if (isScript) {
  4481. runScriptMutation.mutate({ id: script.id, action: 'on' });
  4482. } else {
  4483. setHaToggleConfirm(script);
  4484. }
  4485. }}
  4486. disabled={runScriptMutation.isPending}
  4487. title={`${isScript ? 'Run' : 'Toggle'} ${script.ha_entity_id}`}
  4488. className="px-2 py-0.5 text-xs bg-blue-500/20 text-blue-400 hover:bg-blue-500/30 rounded transition-colors flex items-center gap-1"
  4489. >
  4490. <Play className="w-2.5 h-2.5" />
  4491. {script.name}
  4492. </button>
  4493. );
  4494. })}
  4495. </div>
  4496. </div>
  4497. )}
  4498. </div>
  4499. )}
  4500. {/* Connection Info & Actions - hidden in compact mode */}
  4501. {viewMode === 'expanded' && (
  4502. <div className="mt-4 pt-4 border-t border-bambu-dark-tertiary flex items-center justify-end gap-2 flex-wrap">
  4503. {/* Chamber Light */}
  4504. <Button
  4505. variant="secondary"
  4506. size="sm"
  4507. onClick={() => chamberLightMutation.mutate(!status?.chamber_light)}
  4508. disabled={!status?.connected || chamberLightMutation.isPending || !hasPermission('printers:control')}
  4509. title={!hasPermission('printers:control') ? t('printers.permission.noControl') : (status?.chamber_light ? t('printers.chamberLightOff') : t('printers.chamberLightOn'))}
  4510. className={status?.chamber_light ? '!border-yellow-500 !text-yellow-400 hover:!bg-yellow-500/20' : ''}
  4511. >
  4512. <ChamberLight on={status?.chamber_light ?? false} className={`w-4 h-4 ${status?.chamber_light ? 'text-yellow-400' : ''}`} />
  4513. </Button>
  4514. {/* Camera Button */}
  4515. <Button
  4516. variant="secondary"
  4517. size="sm"
  4518. onClick={() => {
  4519. if (cameraViewMode === 'embedded' && onOpenEmbeddedCamera) {
  4520. onOpenEmbeddedCamera(printer.id, printer.name);
  4521. } else {
  4522. // Use saved window state or defaults
  4523. const saved = localStorage.getItem('cameraWindowState');
  4524. const state = saved ? JSON.parse(saved) : { width: 640, height: 400 };
  4525. const features = [
  4526. `width=${state.width}`,
  4527. `height=${state.height}`,
  4528. state.left !== undefined ? `left=${state.left}` : '',
  4529. state.top !== undefined ? `top=${state.top}` : '',
  4530. // No `noopener`: same-origin popup needs opener so the browser
  4531. // copies sessionStorage (auth token) into the new window.
  4532. 'menubar=no,toolbar=no,location=no,status=no',
  4533. ].filter(Boolean).join(',');
  4534. window.open(`/camera/${printer.id}`, `camera-${printer.id}`, features);
  4535. }
  4536. }}
  4537. disabled={!status?.connected || !hasPermission('camera:view')}
  4538. title={!hasPermission('camera:view') ? t('printers.permission.noCamera') : (cameraViewMode === 'embedded' ? t('printers.openCameraOverlay') : t('printers.openCameraWindow'))}
  4539. >
  4540. <Video className="w-4 h-4" />
  4541. </Button>
  4542. {/* Split button: main part toggles detection, chevron opens modal */}
  4543. <div className={`inline-flex rounded-md ${printer.plate_detection_enabled ? 'ring-1 ring-green-500' : ''}`}>
  4544. <Button
  4545. variant="secondary"
  4546. size="sm"
  4547. onClick={handleTogglePlateDetection}
  4548. disabled={!status?.connected || plateDetectionMutation.isPending || !hasPermission('printers:update')}
  4549. title={!hasPermission('printers:update') ? t('printers.plateDetection.noPermission') : (printer.plate_detection_enabled ? t('printers.plateDetection.enabledClick') : t('printers.plateDetection.disabledClick'))}
  4550. className={`!rounded-r-none !border-r-0 ${printer.plate_detection_enabled ? "!border-green-500 !text-green-400 hover:!bg-green-500/20" : ""}`}
  4551. >
  4552. {plateDetectionMutation.isPending ? (
  4553. <Loader2 className="w-4 h-4 animate-spin" />
  4554. ) : (
  4555. <ScanSearch className="w-4 h-4" />
  4556. )}
  4557. </Button>
  4558. <Button
  4559. variant="secondary"
  4560. size="sm"
  4561. onClick={handleOpenPlateManagement}
  4562. disabled={!status?.connected || isCheckingPlate || !hasPermission('printers:update')}
  4563. title={!hasPermission('printers:update') ? t('printers.plateDetection.noPermission') : t('printers.plateDetection.manageCalibration')}
  4564. className={`!rounded-l-none !px-1.5 ${printer.plate_detection_enabled ? "!border-green-500 !text-green-400 hover:!bg-green-500/20" : ""}`}
  4565. >
  4566. {isCheckingPlate ? (
  4567. <Loader2 className="w-3 h-3 animate-spin" />
  4568. ) : (
  4569. <ChevronDown className="w-3 h-3" />
  4570. )}
  4571. </Button>
  4572. </div>
  4573. <Button
  4574. variant="secondary"
  4575. size="sm"
  4576. onClick={() => setShowFileManager(true)}
  4577. disabled={!isConnected || !hasPermission('printers:files')}
  4578. title={!hasPermission('printers:files') ? t('printers.permission.noFiles') : t('printers.browseFiles')}
  4579. >
  4580. <HardDrive className="w-4 h-4" />
  4581. {t('printers.files')}
  4582. </Button>
  4583. {isConnected && status?.state !== 'RUNNING' && status?.state !== 'PAUSE' && (
  4584. <Button
  4585. size="sm"
  4586. onClick={() => setShowUploadForPrint(true)}
  4587. disabled={!hasPermission('printers:control')}
  4588. title={!hasPermission('printers:control') ? t('printers.permission.noControl') : t('common.print')}
  4589. className="!bg-bambu-green hover:!bg-bambu-green/80 !text-white"
  4590. >
  4591. <PrinterIcon className="w-4 h-4" />
  4592. {t('common.print')}
  4593. </Button>
  4594. )}
  4595. </div>
  4596. )}
  4597. </CardContent>
  4598. {/* File Manager Modal */}
  4599. {showFileManager && (
  4600. <FileManagerModal
  4601. printerId={printer.id}
  4602. printerName={printer.name}
  4603. onClose={() => setShowFileManager(false)}
  4604. />
  4605. )}
  4606. {/* Upload for Print Modal */}
  4607. {showUploadForPrint && (
  4608. <FileUploadModal
  4609. folderId={null}
  4610. onClose={() => setShowUploadForPrint(false)}
  4611. onUploadComplete={() => {}}
  4612. autoUpload
  4613. accept=".gcode,.3mf"
  4614. validateFile={(file) => {
  4615. const lower = file.name.toLowerCase();
  4616. if (!lower.endsWith('.gcode') && !lower.includes('.gcode.')) {
  4617. return t('printers.dropNotPrintable', 'Only .gcode and .gcode.3mf files can be printed');
  4618. }
  4619. }}
  4620. onFileUploaded={(uploadedFile) => {
  4621. // Check printer compatibility if sliced_for_model is available in metadata
  4622. const slicedFor = (uploadedFile.metadata as Record<string, unknown>)?.sliced_for_model as string | undefined;
  4623. const printerModel = mapModelCode(printer.model);
  4624. if (slicedFor && printerModel && slicedFor.toLowerCase() !== printerModel.toLowerCase()) {
  4625. api.deleteLibraryFile(uploadedFile.id).catch(() => {});
  4626. return t('printers.incompatibleFile', 'This file was sliced for {{slicedFor}}, but this printer is a {{printerModel}}', { slicedFor, printerModel });
  4627. }
  4628. setPrintAfterUpload({ id: uploadedFile.id, filename: uploadedFile.filename });
  4629. }}
  4630. />
  4631. )}
  4632. {/* Print Modal (after upload) */}
  4633. {printAfterUpload && (
  4634. <PrintModal
  4635. mode="reprint"
  4636. libraryFileId={printAfterUpload.id}
  4637. archiveName={printAfterUpload.filename}
  4638. initialSelectedPrinterIds={[printer.id]}
  4639. onClose={() => setPrintAfterUpload(null)}
  4640. onSuccess={() => setPrintAfterUpload(null)}
  4641. cleanupLibraryAfterDispatch
  4642. />
  4643. )}
  4644. {/* MQTT Debug Modal */}
  4645. {showMQTTDebug && (
  4646. <MQTTDebugModal
  4647. printerId={printer.id}
  4648. printerName={printer.name}
  4649. onClose={() => setShowMQTTDebug(false)}
  4650. />
  4651. )}
  4652. {showDiagnostic && (
  4653. <ConnectionDiagnosticModal
  4654. printerId={printer.id}
  4655. printerName={printer.name}
  4656. onClose={() => setShowDiagnostic(false)}
  4657. />
  4658. )}
  4659. {showPrinterInfo && (
  4660. <PrinterInfoModal
  4661. printer={printer}
  4662. status={status}
  4663. totalPrintHours={maintenanceInfo?.total_print_hours}
  4664. onClose={closePrinterInfo}
  4665. />
  4666. )}
  4667. {/* Plate Check Result Modal */}
  4668. {plateCheckResult && (
  4669. <div className="fixed inset-0 bg-black/50 flex items-center justify-center z-50 p-4" onClick={() => closePlateCheckModal()}>
  4670. <div className="bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-xl shadow-2xl max-w-lg w-full" onClick={e => e.stopPropagation()}>
  4671. <div className="flex items-center justify-between p-4 border-b border-bambu-dark-tertiary">
  4672. <div className="flex items-center gap-2">
  4673. {plateCheckResult.needs_calibration ? (
  4674. <ScanSearch className="w-5 h-5 text-blue-500" />
  4675. ) : plateCheckResult.is_empty ? (
  4676. <CheckCircle className="w-5 h-5 text-green-500" />
  4677. ) : (
  4678. <XCircle className="w-5 h-5 text-yellow-500" />
  4679. )}
  4680. <h2 className="text-lg font-semibold text-white">
  4681. Build Plate Check
  4682. </h2>
  4683. {plateCheckResult.reference_count !== undefined && plateCheckResult.max_references && (
  4684. <span className="text-xs text-bambu-gray bg-bambu-dark-tertiary px-2 py-1 rounded">
  4685. {plateCheckResult.reference_count}/{plateCheckResult.max_references} refs
  4686. </span>
  4687. )}
  4688. </div>
  4689. <button
  4690. onClick={() => closePlateCheckModal()}
  4691. className="p-1 text-bambu-gray hover:text-white rounded transition-colors"
  4692. >
  4693. <X className="w-5 h-5" />
  4694. </button>
  4695. </div>
  4696. <div className="p-4 space-y-4">
  4697. {plateCheckResult.needs_calibration ? (
  4698. <>
  4699. <div className="p-3 rounded-lg bg-blue-500/20 border border-blue-500/50">
  4700. <p className="font-medium text-blue-400">
  4701. {t('printers.plateDetection.calibrationRequired')}
  4702. </p>
  4703. <p className="text-sm text-bambu-gray mt-1" dangerouslySetInnerHTML={{ __html: t('printers.plateDetection.calibrationInstructions') }} />
  4704. </div>
  4705. <div className="text-sm text-bambu-gray space-y-2">
  4706. <p>{t('printers.plateDetection.calibrationDescription')}</p>
  4707. <p dangerouslySetInnerHTML={{ __html: t('printers.plateDetection.calibrationTip') }} />
  4708. </div>
  4709. </>
  4710. ) : (
  4711. <>
  4712. <div className={`p-3 rounded-lg ${plateCheckResult.is_empty ? 'bg-green-500/20 border border-green-500/50' : 'bg-yellow-500/20 border border-yellow-500/50'}`}>
  4713. <p className={`font-medium ${plateCheckResult.is_empty ? 'text-green-400' : 'text-yellow-400'}`}>
  4714. {plateCheckResult.is_empty ? t('printers.plateDetection.plateEmpty') : t('printers.plateDetection.objectsDetected')}
  4715. </p>
  4716. <p className="text-sm text-bambu-gray mt-1">
  4717. {t('printers.plateDetection.confidence')}: {Math.round(plateCheckResult.confidence * 100)}% | {t('printers.plateDetection.difference')}: {plateCheckResult.difference_percent.toFixed(1)}%
  4718. </p>
  4719. </div>
  4720. {plateCheckResult.debug_image_url && (
  4721. <div>
  4722. <p className="text-sm text-bambu-gray mb-2">{t('printers.plateDetection.analysisPreview')}</p>
  4723. <img
  4724. src={plateCheckResult.debug_image_url}
  4725. alt={t('printers.plateDetection.analysisPreview')}
  4726. className="w-full rounded-lg border border-bambu-dark-tertiary"
  4727. />
  4728. <p className="text-xs text-bambu-gray mt-2">
  4729. {t('printers.plateDetection.analysisLegend')}
  4730. </p>
  4731. </div>
  4732. )}
  4733. <p className="text-xs text-bambu-gray">
  4734. {plateCheckResult.message}
  4735. </p>
  4736. </>
  4737. )}
  4738. {/* Saved References Grid */}
  4739. {plateReferences && plateReferences.references.length > 0 && (
  4740. <div className="mt-4 pt-4 border-t border-bambu-dark-tertiary">
  4741. <p className="text-sm font-medium text-white mb-2">
  4742. {t('printers.plateDetection.savedReferences', { count: plateReferences.references.length, max: plateReferences.max_references })}
  4743. </p>
  4744. <div className="grid grid-cols-5 gap-2">
  4745. {plateReferences.references.map((ref) => (
  4746. <div key={ref.index} className="relative group">
  4747. <img
  4748. src={api.getPlateReferenceThumbnailUrl(printer.id, ref.index)}
  4749. alt={ref.label || `Reference ${ref.index + 1}`}
  4750. className="w-full aspect-video object-cover rounded border border-bambu-dark-tertiary"
  4751. />
  4752. {/* Delete button */}
  4753. <button
  4754. onClick={() => handleDeleteRef(ref.index)}
  4755. className="absolute top-1 right-1 p-0.5 bg-red-500/80 rounded opacity-0 group-hover:opacity-100 transition-opacity"
  4756. title={t('printers.plateDetection.deleteReference')}
  4757. >
  4758. <X className="w-3 h-3 text-white" />
  4759. </button>
  4760. {/* Label */}
  4761. {editingRefLabel?.index === ref.index ? (
  4762. <input
  4763. type="text"
  4764. value={editingRefLabel.label}
  4765. onChange={(e) => setEditingRefLabel({ ...editingRefLabel, label: e.target.value })}
  4766. onBlur={() => handleUpdateRefLabel(ref.index, editingRefLabel.label)}
  4767. onKeyDown={(e) => {
  4768. if (e.key === 'Enter') handleUpdateRefLabel(ref.index, editingRefLabel.label);
  4769. if (e.key === 'Escape') setEditingRefLabel(null);
  4770. }}
  4771. className="w-full mt-1 px-1 py-0.5 text-xs bg-bambu-dark-tertiary border border-bambu-green rounded text-white"
  4772. autoFocus
  4773. placeholder={t('printers.plateDetection.labelPlaceholder')}
  4774. />
  4775. ) : (
  4776. <p
  4777. className="text-xs text-bambu-gray mt-1 truncate cursor-pointer hover:text-white"
  4778. onClick={() => setEditingRefLabel({ index: ref.index, label: ref.label })}
  4779. title={ref.label ? t('printers.plateDetection.clickToEdit', { label: ref.label }) : t('printers.plateDetection.clickToAddLabel')}
  4780. >
  4781. {ref.label || <span className="italic opacity-50">{t('printers.noLabel')}</span>}
  4782. </p>
  4783. )}
  4784. {/* Timestamp */}
  4785. <p className="text-[10px] text-bambu-gray/60">
  4786. {ref.timestamp ? parseUTCDate(ref.timestamp)?.toLocaleDateString() ?? '' : ''}
  4787. </p>
  4788. </div>
  4789. ))}
  4790. </div>
  4791. </div>
  4792. )}
  4793. {/* ROI Editor */}
  4794. {!plateCheckResult.needs_calibration && (
  4795. <div className="mt-4 pt-4 border-t border-bambu-dark-tertiary">
  4796. <div className="flex items-center justify-between mb-2">
  4797. <p className="text-sm font-medium text-white">{t('printers.roi.title')}</p>
  4798. {!editingRoi ? (
  4799. <Button
  4800. variant="ghost"
  4801. size="sm"
  4802. onClick={() => setEditingRoi(plateCheckResult.roi || { x: 0.15, y: 0.35, w: 0.70, h: 0.55 })}
  4803. >
  4804. <Pencil className="w-3 h-3 mr-1" />
  4805. {t('common.edit')}
  4806. </Button>
  4807. ) : (
  4808. <div className="flex gap-1">
  4809. <Button
  4810. variant="ghost"
  4811. size="sm"
  4812. onClick={() => setEditingRoi(null)}
  4813. disabled={isSavingRoi}
  4814. >
  4815. {t('common.cancel')}
  4816. </Button>
  4817. <Button
  4818. size="sm"
  4819. onClick={handleSaveRoi}
  4820. disabled={isSavingRoi}
  4821. >
  4822. {isSavingRoi ? <Loader2 className="w-3 h-3 animate-spin" /> : t('common.save')}
  4823. </Button>
  4824. </div>
  4825. )}
  4826. </div>
  4827. {editingRoi ? (
  4828. <div className="space-y-3 bg-bambu-dark-tertiary/50 p-3 rounded-lg">
  4829. <div className="grid grid-cols-2 gap-3">
  4830. <div>
  4831. <label className="text-xs text-bambu-gray">{t('printers.roi.xStart')}</label>
  4832. <input
  4833. type="range"
  4834. min="0"
  4835. max="0.9"
  4836. step="0.01"
  4837. value={editingRoi.x}
  4838. onChange={(e) => setEditingRoi({ ...editingRoi, x: parseFloat(e.target.value) })}
  4839. className="w-full h-1.5 bg-bambu-dark-tertiary rounded-lg cursor-pointer accent-green-500"
  4840. />
  4841. <span className="text-xs text-bambu-gray">{Math.round(editingRoi.x * 100)}%</span>
  4842. </div>
  4843. <div>
  4844. <label className="text-xs text-bambu-gray">{t('printers.roi.yStart')}</label>
  4845. <input
  4846. type="range"
  4847. min="0"
  4848. max="0.9"
  4849. step="0.01"
  4850. value={editingRoi.y}
  4851. onChange={(e) => setEditingRoi({ ...editingRoi, y: parseFloat(e.target.value) })}
  4852. className="w-full h-1.5 bg-bambu-dark-tertiary rounded-lg cursor-pointer accent-green-500"
  4853. />
  4854. <span className="text-xs text-bambu-gray">{Math.round(editingRoi.y * 100)}%</span>
  4855. </div>
  4856. <div>
  4857. <label className="text-xs text-bambu-gray">{t('printers.width')}</label>
  4858. <input
  4859. type="range"
  4860. min="0.1"
  4861. max="1"
  4862. step="0.01"
  4863. value={editingRoi.w}
  4864. onChange={(e) => setEditingRoi({ ...editingRoi, w: parseFloat(e.target.value) })}
  4865. className="w-full h-1.5 bg-bambu-dark-tertiary rounded-lg cursor-pointer accent-green-500"
  4866. />
  4867. <span className="text-xs text-bambu-gray">{Math.round(editingRoi.w * 100)}%</span>
  4868. </div>
  4869. <div>
  4870. <label className="text-xs text-bambu-gray">{t('printers.height')}</label>
  4871. <input
  4872. type="range"
  4873. min="0.1"
  4874. max="1"
  4875. step="0.01"
  4876. value={editingRoi.h}
  4877. onChange={(e) => setEditingRoi({ ...editingRoi, h: parseFloat(e.target.value) })}
  4878. className="w-full h-1.5 bg-bambu-dark-tertiary rounded-lg cursor-pointer accent-green-500"
  4879. />
  4880. <span className="text-xs text-bambu-gray">{Math.round(editingRoi.h * 100)}%</span>
  4881. </div>
  4882. </div>
  4883. <p className="text-xs text-bambu-gray">
  4884. {t('printers.roi.instruction')}
  4885. </p>
  4886. </div>
  4887. ) : (
  4888. <p className="text-xs text-bambu-gray">
  4889. Current: X={Math.round((plateCheckResult.roi?.x || 0.15) * 100)}%, Y={Math.round((plateCheckResult.roi?.y || 0.35) * 100)}%,
  4890. W={Math.round((plateCheckResult.roi?.w || 0.70) * 100)}%, H={Math.round((plateCheckResult.roi?.h || 0.55) * 100)}%
  4891. </p>
  4892. )}
  4893. </div>
  4894. )}
  4895. </div>
  4896. <div className="flex justify-end gap-2 p-4 border-t border-bambu-dark-tertiary">
  4897. {plateCheckResult.needs_calibration ? (
  4898. <>
  4899. <Button variant="ghost" onClick={() => closePlateCheckModal()}>
  4900. {t('common.cancel')}
  4901. </Button>
  4902. <Button
  4903. onClick={() => handleCalibratePlate()}
  4904. disabled={isCalibrating}
  4905. >
  4906. {isCalibrating ? (
  4907. <>
  4908. <Loader2 className="w-4 h-4 mr-2 animate-spin" />
  4909. Calibrating...
  4910. </>
  4911. ) : (
  4912. 'Calibrate Empty Plate'
  4913. )}
  4914. </Button>
  4915. </>
  4916. ) : (
  4917. <>
  4918. <Button variant="ghost" onClick={() => handleCalibratePlate()} disabled={isCalibrating}>
  4919. {isCalibrating ? 'Adding...' : `Add Reference (${plateReferences?.references.length || 0}/${plateReferences?.max_references || 5})`}
  4920. </Button>
  4921. <Button onClick={() => closePlateCheckModal()}>
  4922. Close
  4923. </Button>
  4924. </>
  4925. )}
  4926. </div>
  4927. </div>
  4928. </div>
  4929. )}
  4930. {/* Power On Confirmation */}
  4931. {showPowerOnConfirm && smartPlug && (
  4932. <ConfirmModal
  4933. title={t('printers.confirm.powerOnTitle')}
  4934. message={t('printers.confirm.powerOnMessage', { name: printer.name })}
  4935. confirmText={t('printers.confirm.powerOnButton')}
  4936. variant="default"
  4937. onConfirm={() => {
  4938. powerControlMutation.mutate('on');
  4939. setShowPowerOnConfirm(false);
  4940. }}
  4941. onCancel={() => setShowPowerOnConfirm(false)}
  4942. />
  4943. )}
  4944. {/* Power Off Confirmation */}
  4945. {showPowerOffConfirm && smartPlug && (
  4946. <ConfirmModal
  4947. title={t('printers.confirm.powerOffTitle')}
  4948. message={
  4949. status?.state === 'RUNNING'
  4950. ? t('printers.confirm.powerOffWarning', { name: printer.name })
  4951. : t('printers.confirm.powerOffMessage', { name: printer.name })
  4952. }
  4953. confirmText={t('printers.confirm.powerOffButton')}
  4954. variant="danger"
  4955. onConfirm={() => {
  4956. powerControlMutation.mutate('off');
  4957. setShowPowerOffConfirm(false);
  4958. }}
  4959. onCancel={() => setShowPowerOffConfirm(false)}
  4960. />
  4961. )}
  4962. {/* HA entity toggle confirmation (Show on Printer Card switches) */}
  4963. {haToggleConfirm && (
  4964. <ConfirmModal
  4965. title={t('printers.confirm.haToggleTitle', { name: haToggleConfirm.name })}
  4966. message={
  4967. status?.state === 'RUNNING'
  4968. ? t('printers.confirm.haToggleWarning', { name: printer.name, entity: haToggleConfirm.ha_entity_id || haToggleConfirm.name })
  4969. : t('printers.confirm.haToggleMessage', { entity: haToggleConfirm.ha_entity_id || haToggleConfirm.name })
  4970. }
  4971. confirmText={t('printers.confirm.haToggleButton')}
  4972. variant={status?.state === 'RUNNING' ? 'danger' : 'default'}
  4973. onConfirm={() => {
  4974. runScriptMutation.mutate({ id: haToggleConfirm.id, action: 'toggle' });
  4975. setHaToggleConfirm(null);
  4976. }}
  4977. onCancel={() => setHaToggleConfirm(null)}
  4978. />
  4979. )}
  4980. {/* Stop Print Confirmation */}
  4981. {showStopConfirm && (
  4982. <ConfirmModal
  4983. title={t('printers.confirm.stopTitle')}
  4984. message={t('printers.confirm.stopMessage', { name: printer.name })}
  4985. confirmText={t('printers.confirm.stopButton')}
  4986. variant="danger"
  4987. onConfirm={() => {
  4988. stopPrintMutation.mutate();
  4989. setShowStopConfirm(false);
  4990. }}
  4991. onCancel={() => setShowStopConfirm(false)}
  4992. />
  4993. )}
  4994. {/* Pause Print Confirmation */}
  4995. {showPauseConfirm && (
  4996. <ConfirmModal
  4997. title={t('printers.confirm.pauseTitle')}
  4998. message={t('printers.confirm.pauseMessage', { name: printer.name })}
  4999. confirmText={t('printers.confirm.pauseButton')}
  5000. variant="default"
  5001. onConfirm={() => {
  5002. pausePrintMutation.mutate();
  5003. setShowPauseConfirm(false);
  5004. }}
  5005. onCancel={() => setShowPauseConfirm(false)}
  5006. />
  5007. )}
  5008. {/* Resume Print Confirmation */}
  5009. {showResumeConfirm && (
  5010. <ConfirmModal
  5011. title={t('printers.confirm.resumeTitle')}
  5012. message={t('printers.confirm.resumeMessage', { name: printer.name })}
  5013. confirmText={t('printers.confirm.resumeButton')}
  5014. variant="default"
  5015. onConfirm={() => {
  5016. resumePrintMutation.mutate();
  5017. setShowResumeConfirm(false);
  5018. }}
  5019. onCancel={() => setShowResumeConfirm(false)}
  5020. />
  5021. )}
  5022. {/* Bed Jog — not-homed warning (Studio-style) */}
  5023. {showNotHomedModal && (
  5024. <div className="fixed inset-0 z-50 flex items-center justify-center bg-black/70 p-4">
  5025. <div className="bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg shadow-xl w-full max-w-sm p-5">
  5026. <div className="flex items-start gap-3 mb-4">
  5027. <AlertTriangle className="w-5 h-5 text-yellow-400 flex-shrink-0 mt-0.5" />
  5028. <div>
  5029. <h3 className="text-sm font-semibold text-white mb-1">
  5030. {t('printers.bedJog.notHomedTitle')}
  5031. </h3>
  5032. <p className="text-xs text-bambu-gray leading-relaxed">
  5033. {t('printers.bedJog.notHomedMessage')}
  5034. </p>
  5035. </div>
  5036. </div>
  5037. <div className="flex flex-col gap-2">
  5038. <button
  5039. onClick={() => {
  5040. homeAxesMutation.mutate('all');
  5041. setShowNotHomedModal(null);
  5042. }}
  5043. className="w-full px-3 py-2 rounded-lg text-xs font-medium bg-bambu-green/20 text-bambu-green hover:bg-bambu-green/30 transition-colors"
  5044. >
  5045. {t('printers.bedJog.homeZ')}
  5046. </button>
  5047. <button
  5048. onClick={() => {
  5049. const d = showNotHomedModal.distance;
  5050. try { sessionStorage.setItem(`bambuddy.bedJog.warned.${printer.id}`, '1'); } catch { /* ignore */ }
  5051. bedJogMutation.mutate({ distance: d, force: true });
  5052. setShowNotHomedModal(null);
  5053. }}
  5054. className="w-full px-3 py-2 rounded-lg text-xs font-medium bg-yellow-500/20 text-yellow-400 hover:bg-yellow-500/30 transition-colors"
  5055. >
  5056. {t('printers.bedJog.moveAnyway')}
  5057. </button>
  5058. <button
  5059. onClick={() => setShowNotHomedModal(null)}
  5060. className="w-full px-3 py-2 rounded-lg text-xs font-medium bg-bambu-dark text-bambu-gray hover:bg-bambu-dark-tertiary transition-colors"
  5061. >
  5062. {t('common.cancel')}
  5063. </button>
  5064. </div>
  5065. </div>
  5066. </div>
  5067. )}
  5068. {/* Skip Objects Modal */}
  5069. <SkipObjectsModal
  5070. printerId={printer.id}
  5071. isOpen={showSkipObjectsModal}
  5072. onClose={() => setShowSkipObjectsModal(false)}
  5073. />
  5074. {/* HMS Error Modal */}
  5075. {showHMSModal && (
  5076. <HMSErrorModal
  5077. printerName={printer.name}
  5078. errors={status?.hms_errors || []}
  5079. onClose={() => setShowHMSModal(false)}
  5080. printerId={printer.id}
  5081. hasPermission={hasPermission}
  5082. />
  5083. )}
  5084. {/* AMS History Modal */}
  5085. {amsHistoryModal && (
  5086. <AMSHistoryModal
  5087. isOpen={!!amsHistoryModal}
  5088. onClose={() => setAmsHistoryModal(null)}
  5089. printerId={printer.id}
  5090. printerName={printer.name}
  5091. amsId={amsHistoryModal.amsId}
  5092. amsLabel={amsHistoryModal.amsLabel}
  5093. initialMode={amsHistoryModal.mode}
  5094. thresholds={amsThresholds}
  5095. />
  5096. )}
  5097. {/* Link Spool Modal */}
  5098. {linkSpoolModal && (
  5099. <LinkSpoolModal
  5100. isOpen={!!linkSpoolModal}
  5101. onClose={() => setLinkSpoolModal(null)}
  5102. tagUid={linkSpoolModal.tagUid}
  5103. trayUuid={linkSpoolModal.trayUuid}
  5104. printerId={linkSpoolModal.printerId}
  5105. amsId={linkSpoolModal.amsId}
  5106. trayId={linkSpoolModal.trayId}
  5107. />
  5108. )}
  5109. {/* Assign Spool Modal */}
  5110. {assignSpoolModal && (
  5111. <AssignSpoolModal
  5112. isOpen={!!assignSpoolModal}
  5113. onClose={() => setAssignSpoolModal(null)}
  5114. printerId={assignSpoolModal.printerId}
  5115. amsId={assignSpoolModal.amsId}
  5116. trayId={assignSpoolModal.trayId}
  5117. trayInfo={assignSpoolModal.trayInfo}
  5118. spoolmanEnabled={!!spoolmanEnabled}
  5119. />
  5120. )}
  5121. {/* Configure AMS Slot Modal */}
  5122. {configureSlotModal && (
  5123. <ConfigureAmsSlotModal
  5124. isOpen={!!configureSlotModal}
  5125. onClose={() => setConfigureSlotModal(null)}
  5126. printerId={printer.id}
  5127. slotInfo={configureSlotModal}
  5128. printerModel={mapModelCode(printer.model) || undefined}
  5129. onSuccess={() => {
  5130. // Refresh slot presets to show updated profile name
  5131. queryClient.invalidateQueries({ queryKey: ['slotPresets', printer.id] });
  5132. // Printer status will update automatically via WebSocket when AMS data changes
  5133. queryClient.invalidateQueries({ queryKey: ['printerStatus', printer.id] });
  5134. }}
  5135. />
  5136. )}
  5137. {/* Edit Printer Modal */}
  5138. {showEditModal && (
  5139. <EditPrinterModal
  5140. printer={printer}
  5141. onClose={() => setShowEditModal(false)}
  5142. />
  5143. )}
  5144. {/* Firmware Update Modal */}
  5145. {showFirmwareModal && firmwareInfo && (
  5146. <FirmwareUpdateModal
  5147. printer={printer}
  5148. firmwareInfo={firmwareInfo}
  5149. onClose={() => setShowFirmwareModal(false)}
  5150. />
  5151. )}
  5152. {/* AMS Slot Menu Backdrop - closes menu when clicking outside */}
  5153. {amsSlotMenu && (
  5154. <div
  5155. className="fixed inset-0 z-40"
  5156. onClick={() => setAmsSlotMenu(null)}
  5157. />
  5158. )}
  5159. {/* AMS Drying Popover — fixed position to avoid overflow/z-index issues */}
  5160. {dryingPopoverAmsId !== null && dryingPopoverPos && (() => {
  5161. const maxTemp = dryingPopoverModuleType === 'n3s' ? 85 : 65;
  5162. const sliderMin = 35;
  5163. const sliderMax = maxTemp + 10;
  5164. return (
  5165. <>
  5166. {/* Backdrop */}
  5167. <div className="fixed inset-0 z-[100]" onClick={() => setDryingPopoverAmsId(null)} />
  5168. {/* Popover */}
  5169. <div
  5170. className="fixed z-[101] flex flex-col w-[240px] bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-xl shadow-2xl overflow-hidden"
  5171. style={{
  5172. top: dryingPopoverPos.top,
  5173. left: dryingPopoverPos.left,
  5174. // Cap to the space between the popover's top and the bottom
  5175. // viewport margin (8px, matching computePopoverPosition's
  5176. // margin). When the popover is taller than that space — short
  5177. // viewport, landscape phone, zoomed-in — the body scrolls and
  5178. // the footer stays pinned, so the Start button is always
  5179. // reachable (#1458 / #1447 follow-up).
  5180. maxHeight: `calc(100vh - ${dryingPopoverPos.top}px - 8px)`,
  5181. }}
  5182. onClick={e => e.stopPropagation()}
  5183. >
  5184. {/* Header */}
  5185. <div className="shrink-0 flex items-center gap-2 px-3 py-2.5 border-b border-bambu-dark-tertiary">
  5186. <Flame className="w-3.5 h-3.5 text-amber-400" />
  5187. <span className="text-xs text-white font-medium">{t('printers.drying.start')}</span>
  5188. </div>
  5189. {/* Body */}
  5190. <div className="px-3 py-2.5 space-y-2.5 overflow-y-auto min-h-0">
  5191. {/* Filament type select */}
  5192. <div>
  5193. <label className="text-[10px] text-bambu-gray mb-1 block">{t('printers.filaments')}</label>
  5194. <select
  5195. value={dryingFilament}
  5196. onChange={e => {
  5197. const fil = e.target.value;
  5198. setDryingFilament(fil);
  5199. const preset = dryingPresets[fil];
  5200. if (preset) {
  5201. const key = dryingPopoverModuleType === 'n3s' ? 'n3s' : 'n3f';
  5202. setDryingTemp(preset[key]);
  5203. setDryingDuration(dryingPopoverModuleType === 'n3s' ? preset.n3s_hours : preset.n3f_hours);
  5204. }
  5205. }}
  5206. className="w-full px-2 py-1.5 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white text-xs focus:outline-none focus:border-amber-500/50"
  5207. >
  5208. {Object.keys(dryingPresets).map(fil => (
  5209. <option key={fil} value={fil}>{fil}</option>
  5210. ))}
  5211. </select>
  5212. </div>
  5213. {/* Temperature */}
  5214. <div>
  5215. <div className="flex items-center justify-between mb-1">
  5216. <label className="text-[10px] text-bambu-gray">{t('printers.drying.temperature')}</label>
  5217. <div className="flex items-center gap-1">
  5218. <input
  5219. type="number"
  5220. min={45}
  5221. max={maxTemp}
  5222. value={dryingTemp}
  5223. onChange={e => setDryingTemp(Math.min(maxTemp, Math.max(45, Number(e.target.value) || 45)))}
  5224. className="w-12 px-1 py-0.5 bg-bambu-dark border border-bambu-dark-tertiary rounded text-white text-[11px] text-center focus:outline-none focus:border-amber-500/50 [appearance:textfield] [&::-webkit-outer-spin-button]:appearance-none [&::-webkit-inner-spin-button]:appearance-none"
  5225. />
  5226. <span className="text-[10px] text-bambu-gray">°C</span>
  5227. </div>
  5228. </div>
  5229. <input
  5230. type="range"
  5231. min={sliderMin}
  5232. max={sliderMax}
  5233. value={dryingTemp}
  5234. onChange={e => setDryingTemp(Math.min(maxTemp, Math.max(45, Number(e.target.value))))}
  5235. className="w-full h-1 accent-amber-500 cursor-pointer"
  5236. />
  5237. <div className="flex justify-between text-[9px] text-bambu-gray/50 mt-0.5">
  5238. <span>45°C</span>
  5239. <span>{maxTemp}°C</span>
  5240. </div>
  5241. </div>
  5242. {/* Duration */}
  5243. <div>
  5244. <div className="flex items-center justify-between mb-1">
  5245. <label className="text-[10px] text-bambu-gray">{t('printers.drying.duration')}</label>
  5246. <div className="flex items-center gap-1">
  5247. <input
  5248. type="number"
  5249. min={1}
  5250. max={24}
  5251. value={dryingDuration}
  5252. onChange={e => setDryingDuration(Math.min(24, Math.max(1, Number(e.target.value) || 1)))}
  5253. className="w-10 px-1 py-0.5 bg-bambu-dark border border-bambu-dark-tertiary rounded text-white text-[11px] text-center focus:outline-none focus:border-amber-500/50 [appearance:textfield] [&::-webkit-outer-spin-button]:appearance-none [&::-webkit-inner-spin-button]:appearance-none"
  5254. />
  5255. <span className="text-[10px] text-bambu-gray">{t('printers.drying.hours')}</span>
  5256. </div>
  5257. </div>
  5258. <input
  5259. type="range"
  5260. min={1}
  5261. max={24}
  5262. value={dryingDuration}
  5263. onChange={e => setDryingDuration(Number(e.target.value))}
  5264. className="w-full h-1 accent-amber-500 cursor-pointer"
  5265. />
  5266. <div className="flex justify-between text-[9px] text-bambu-gray/50 mt-0.5">
  5267. <span>1h</span>
  5268. <span>24h</span>
  5269. </div>
  5270. </div>
  5271. {/* Rotate tray */}
  5272. <label className="flex items-center gap-2 cursor-pointer">
  5273. <input
  5274. type="checkbox"
  5275. checked={dryingRotateTray}
  5276. onChange={e => setDryingRotateTray(e.target.checked)}
  5277. className="w-3.5 h-3.5 accent-amber-500 rounded cursor-pointer"
  5278. />
  5279. <span className="text-[11px] text-bambu-gray">{t('printers.drying.rotateTray')}</span>
  5280. </label>
  5281. </div>
  5282. {/* Footer */}
  5283. <div className="shrink-0 px-3 pt-2.5 pb-3">
  5284. <button
  5285. onClick={() => {
  5286. if (dryingPopoverAmsId !== null) {
  5287. startDryingMutation.mutate({ amsId: dryingPopoverAmsId, temp: dryingTemp, duration: dryingDuration, filament: dryingFilament, rotateTray: dryingRotateTray });
  5288. }
  5289. }}
  5290. disabled={startDryingMutation.isPending}
  5291. className="w-full py-1.5 bg-amber-500 hover:bg-amber-400 text-white text-xs font-medium rounded-lg transition-colors disabled:opacity-50"
  5292. >
  5293. {startDryingMutation.isPending ? t('printers.drying.startingDrying') : t('printers.drying.start')}
  5294. </button>
  5295. </div>
  5296. </div>
  5297. </>
  5298. );
  5299. })()}
  5300. </Card>
  5301. );
  5302. }
  5303. function AddPrinterModal({
  5304. onClose,
  5305. onAdd,
  5306. existingSerials,
  5307. }: {
  5308. onClose: () => void;
  5309. onAdd: (data: PrinterCreate) => void;
  5310. existingSerials: string[];
  5311. }) {
  5312. const { t } = useTranslation();
  5313. const [form, setForm] = useState<PrinterCreate>({
  5314. name: '',
  5315. serial_number: '',
  5316. ip_address: '',
  5317. access_code: '',
  5318. model: '',
  5319. location: '',
  5320. auto_archive: true,
  5321. });
  5322. // Discovery state
  5323. const [discovering, setDiscovering] = useState(false);
  5324. const [discovered, setDiscovered] = useState<DiscoveredPrinter[]>([]);
  5325. const [discoveryError, setDiscoveryError] = useState('');
  5326. const [hasScanned, setHasScanned] = useState(false);
  5327. const [isDocker, setIsDocker] = useState(false);
  5328. const [detectedSubnets, setDetectedSubnets] = useState<string[]>([]);
  5329. const [subnet, setSubnet] = useState('');
  5330. const [scanProgress, setScanProgress] = useState({ scanned: 0, total: 0 });
  5331. const [showDiagnostic, setShowDiagnostic] = useState(false);
  5332. // Setup-time pre-flight: run the connection diagnostic on save and warn
  5333. // (not block) when checks fail, so the user doesn't add a printer that
  5334. // immediately shows offline. checkingSave = probe in flight; saveWarning =
  5335. // failed result awaiting an explicit "save anyway".
  5336. const [checkingSave, setCheckingSave] = useState(false);
  5337. const [saveWarning, setSaveWarning] = useState<PrinterDiagnosticResult | null>(null);
  5338. // Fetch discovery info on mount
  5339. useEffect(() => {
  5340. discoveryApi.getInfo().then(info => {
  5341. setIsDocker(info.is_docker);
  5342. if (info.subnets.length > 0) {
  5343. setDetectedSubnets(info.subnets);
  5344. setSubnet(info.subnets[0]);
  5345. }
  5346. }).catch(() => {
  5347. // Ignore errors, assume not Docker
  5348. });
  5349. }, []);
  5350. // Filter out already-added printers
  5351. const newPrinters = discovered.filter(p => !existingSerials.includes(p.serial));
  5352. const handleAddSubmit = async (e: React.FormEvent) => {
  5353. e.preventDefault();
  5354. setCheckingSave(true);
  5355. try {
  5356. const result = await api.diagnoseConnection({
  5357. ip_address: form.ip_address.trim(),
  5358. serial_number: form.serial_number.trim() || undefined,
  5359. access_code: form.access_code || undefined,
  5360. });
  5361. if (result.checks.some((c) => c.status === 'fail')) {
  5362. setSaveWarning(result);
  5363. return;
  5364. }
  5365. } catch {
  5366. // Diagnostic infrastructure failed — never block the save on it.
  5367. } finally {
  5368. setCheckingSave(false);
  5369. }
  5370. onAdd(form);
  5371. };
  5372. const startDiscovery = async () => {
  5373. setDiscoveryError('');
  5374. setDiscovered([]);
  5375. setDiscovering(true);
  5376. setHasScanned(false);
  5377. setScanProgress({ scanned: 0, total: 0 });
  5378. try {
  5379. if (isDocker) {
  5380. // Use subnet scanning for Docker
  5381. await discoveryApi.startSubnetScan(subnet);
  5382. // Poll for scan status and results
  5383. const pollInterval = setInterval(async () => {
  5384. try {
  5385. const status = await discoveryApi.getScanStatus();
  5386. setScanProgress({ scanned: status.scanned, total: status.total });
  5387. const printers = await discoveryApi.getDiscoveredPrinters();
  5388. setDiscovered(printers);
  5389. if (!status.running) {
  5390. clearInterval(pollInterval);
  5391. setDiscovering(false);
  5392. setHasScanned(true);
  5393. }
  5394. } catch (e) {
  5395. console.error('Failed to get scan status:', e);
  5396. }
  5397. }, 500);
  5398. } else {
  5399. // Use SSDP discovery for native installs
  5400. await discoveryApi.startDiscovery(10);
  5401. // Poll for discovered printers every second
  5402. const pollInterval = setInterval(async () => {
  5403. try {
  5404. const printers = await discoveryApi.getDiscoveredPrinters();
  5405. setDiscovered(printers);
  5406. } catch (e) {
  5407. console.error('Failed to get discovered printers:', e);
  5408. }
  5409. }, 1000);
  5410. // Stop after 10 seconds
  5411. setTimeout(async () => {
  5412. clearInterval(pollInterval);
  5413. try {
  5414. await discoveryApi.stopDiscovery();
  5415. } catch {
  5416. // Ignore stop errors
  5417. }
  5418. setDiscovering(false);
  5419. setHasScanned(true);
  5420. // Final fetch
  5421. try {
  5422. const printers = await discoveryApi.getDiscoveredPrinters();
  5423. setDiscovered(printers);
  5424. } catch (e) {
  5425. console.error('Failed to get final discovered printers:', e);
  5426. }
  5427. }, 10000);
  5428. }
  5429. } catch (e) {
  5430. console.error('Failed to start discovery:', e);
  5431. setDiscoveryError(e instanceof Error ? e.message : t('printers.discovery.failedToStart'));
  5432. setDiscovering(false);
  5433. setHasScanned(true);
  5434. }
  5435. };
  5436. // Reuse module-level mapModelCode
  5437. const selectPrinter = (printer: DiscoveredPrinter) => {
  5438. // Don't pre-fill serial if it's a placeholder (unknown-*) - user needs to enter actual serial
  5439. const serialNumber = printer.serial.startsWith('unknown-') ? '' : printer.serial;
  5440. setForm({
  5441. ...form,
  5442. name: printer.name || '',
  5443. serial_number: serialNumber,
  5444. ip_address: printer.ip_address,
  5445. model: mapModelCode(printer.model),
  5446. });
  5447. // Clear discovery results after selection
  5448. setDiscovered([]);
  5449. };
  5450. // Cleanup discovery on unmount
  5451. useEffect(() => {
  5452. return () => {
  5453. discoveryApi.stopDiscovery().catch(() => {});
  5454. discoveryApi.stopSubnetScan().catch(() => {});
  5455. };
  5456. }, []);
  5457. // Close on Escape key
  5458. useEffect(() => {
  5459. const handleKeyDown = (e: KeyboardEvent) => {
  5460. if (e.key === 'Escape') onClose();
  5461. };
  5462. window.addEventListener('keydown', handleKeyDown);
  5463. return () => window.removeEventListener('keydown', handleKeyDown);
  5464. }, [onClose]);
  5465. return (
  5466. <>
  5467. <div
  5468. className="fixed inset-0 bg-black/50 flex items-start sm:items-center justify-center z-50 p-4 overflow-y-auto"
  5469. onClick={onClose}
  5470. >
  5471. <Card className="w-full max-w-md my-auto max-h-[calc(100vh-2rem)] overflow-y-auto" onClick={(e: React.MouseEvent) => e.stopPropagation()}>
  5472. <CardContent>
  5473. <h2 className="text-xl font-semibold mb-4">{t('printers.addPrinter')}</h2>
  5474. {/* Discovery Section */}
  5475. <div className="mb-4 pb-4 border-b border-bambu-dark-tertiary">
  5476. {isDocker && (
  5477. <div className="mb-3">
  5478. <label className="block text-sm text-bambu-gray mb-1">
  5479. {t('printers.discovery.subnetToScan')}
  5480. </label>
  5481. {detectedSubnets.length > 0 ? (
  5482. <select
  5483. 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 text-sm"
  5484. value={subnet}
  5485. onChange={(e) => setSubnet(e.target.value)}
  5486. disabled={discovering}
  5487. >
  5488. {detectedSubnets.map(s => (
  5489. <option key={s} value={s}>{s}</option>
  5490. ))}
  5491. </select>
  5492. ) : (
  5493. <input
  5494. type="text"
  5495. 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 text-sm"
  5496. value={subnet}
  5497. onChange={(e) => setSubnet(e.target.value)}
  5498. placeholder="192.168.1.0/24"
  5499. disabled={discovering}
  5500. />
  5501. )}
  5502. <p className="mt-1 text-xs text-bambu-gray">
  5503. {t('printers.discovery.dockerNote')}
  5504. </p>
  5505. </div>
  5506. )}
  5507. <Button
  5508. type="button"
  5509. variant="secondary"
  5510. onClick={startDiscovery}
  5511. disabled={discovering}
  5512. className="w-full"
  5513. >
  5514. {discovering ? (
  5515. <>
  5516. <Loader2 className="w-4 h-4 animate-spin" />
  5517. {isDocker && scanProgress.total > 0
  5518. ? t('printers.discovery.scanProgress', { scanned: scanProgress.scanned, total: scanProgress.total })
  5519. : t('printers.discovery.scanning')}
  5520. </>
  5521. ) : (
  5522. <>
  5523. <Search className="w-4 h-4" />
  5524. {isDocker ? t('printers.discovery.scanSubnet') : t('printers.discovery.discoverNetwork')}
  5525. </>
  5526. )}
  5527. </Button>
  5528. {discoveryError && (
  5529. <div className="mt-2 text-sm text-red-400">{discoveryError}</div>
  5530. )}
  5531. {newPrinters.length > 0 && (
  5532. <div className="mt-3 space-y-2 max-h-40 overflow-y-auto">
  5533. {newPrinters.map((printer) => (
  5534. <div
  5535. key={printer.serial}
  5536. className="flex items-center justify-between p-2 bg-bambu-dark rounded-lg hover:bg-bambu-dark-secondary cursor-pointer transition-colors"
  5537. onClick={() => selectPrinter(printer)}
  5538. >
  5539. <div className="min-w-0 flex-1">
  5540. <p className="font-medium text-white text-sm truncate">
  5541. {printer.name || printer.serial}
  5542. </p>
  5543. <p className="text-xs text-bambu-gray truncate">
  5544. {mapModelCode(printer.model) || t('printers.discovery.unknown')} • {printer.ip_address}
  5545. {printer.serial.startsWith('unknown-') && (
  5546. <span className="text-yellow-500"> • {t('printers.discovery.serialRequired')}</span>
  5547. )}
  5548. </p>
  5549. </div>
  5550. <ChevronDown className="w-4 h-4 text-bambu-gray -rotate-90 flex-shrink-0 ml-2" />
  5551. </div>
  5552. ))}
  5553. </div>
  5554. )}
  5555. {discovering && (
  5556. <p className="mt-2 text-sm text-bambu-gray text-center">
  5557. {isDocker ? t('printers.discovery.scanningSubnet') : t('printers.discovery.scanningNetwork')}
  5558. </p>
  5559. )}
  5560. {hasScanned && !discovering && discovered.length === 0 && (
  5561. <p className="mt-2 text-sm text-bambu-gray text-center">
  5562. {isDocker ? t('printers.discovery.noPrintersFoundSubnet') : t('printers.discovery.noPrintersFoundNetwork')}
  5563. </p>
  5564. )}
  5565. {hasScanned && !discovering && discovered.length > 0 && newPrinters.length === 0 && (
  5566. <p className="mt-2 text-sm text-bambu-gray text-center">
  5567. {t('printers.discovery.allConfigured')}
  5568. </p>
  5569. )}
  5570. </div>
  5571. <form onSubmit={handleAddSubmit} className="space-y-4">
  5572. <div>
  5573. <label className="block text-sm text-bambu-gray mb-1">{t('printers.name')}</label>
  5574. <input
  5575. type="text"
  5576. required
  5577. 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"
  5578. value={form.name}
  5579. onChange={(e) => setForm({ ...form, name: e.target.value })}
  5580. placeholder={t('printers.modal.myPrinter')}
  5581. />
  5582. </div>
  5583. <div>
  5584. <label className="block text-sm text-bambu-gray mb-1">{t('printers.ipAddress')}</label>
  5585. <input
  5586. type="text"
  5587. required
  5588. pattern="(\d{1,3}(\.\d{1,3}){3}|[a-zA-Z0-9]([a-zA-Z0-9\-]{0,61}[a-zA-Z0-9])?(\.[a-zA-Z0-9]([a-zA-Z0-9\-]{0,61}[a-zA-Z0-9])?)*)"
  5589. 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"
  5590. value={form.ip_address}
  5591. onChange={(e) => setForm({ ...form, ip_address: e.target.value })}
  5592. placeholder="192.168.1.100 or printer.local"
  5593. />
  5594. </div>
  5595. <div>
  5596. <label className="block text-sm text-bambu-gray mb-1">{t('printers.serialNumber')}</label>
  5597. <input
  5598. type="text"
  5599. required
  5600. 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"
  5601. value={form.serial_number}
  5602. onChange={(e) => setForm({ ...form, serial_number: e.target.value })}
  5603. placeholder="01P00A000000000"
  5604. />
  5605. </div>
  5606. <div>
  5607. <label className="block text-sm text-bambu-gray mb-1">{t('printers.accessCode')}</label>
  5608. <input
  5609. type="password"
  5610. required
  5611. 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"
  5612. value={form.access_code}
  5613. onChange={(e) => setForm({ ...form, access_code: e.target.value })}
  5614. placeholder={t('printers.modal.fromPrinterSettings')}
  5615. />
  5616. </div>
  5617. <div>
  5618. <label className="block text-sm text-bambu-gray mb-1">{t('printers.modal.modelOptional')}</label>
  5619. <select
  5620. 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"
  5621. value={form.model || ''}
  5622. onChange={(e) => setForm({ ...form, model: e.target.value })}
  5623. >
  5624. <option value="">{t('printers.modal.selectModel')}</option>
  5625. <optgroup label="H2 Series">
  5626. <option value="H2C">H2C</option>
  5627. <option value="H2D">H2D</option>
  5628. <option value="H2D Pro">H2D Pro</option>
  5629. <option value="H2S">H2S</option>
  5630. </optgroup>
  5631. <optgroup label="X2 Series">
  5632. <option value="X2D">X2D</option>
  5633. </optgroup>
  5634. <optgroup label="X1 Series">
  5635. <option value="X1E">X1E</option>
  5636. <option value="X1C">X1 Carbon</option>
  5637. <option value="X1">X1</option>
  5638. </optgroup>
  5639. <optgroup label="P Series">
  5640. <option value="P2S">P2S</option>
  5641. <option value="P1S">P1S</option>
  5642. <option value="P1P">P1P</option>
  5643. </optgroup>
  5644. <optgroup label="A1 Series">
  5645. <option value="A1">A1</option>
  5646. <option value="A1 Mini">A1 Mini</option>
  5647. </optgroup>
  5648. </select>
  5649. </div>
  5650. <div>
  5651. <label className="block text-sm text-bambu-gray mb-1">{t('printers.modal.locationGroup')}</label>
  5652. <input
  5653. type="text"
  5654. 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"
  5655. value={form.location || ''}
  5656. onChange={(e) => setForm({ ...form, location: e.target.value })}
  5657. placeholder={t('printers.modal.locationPlaceholder')}
  5658. />
  5659. <p className="text-xs text-bambu-gray mt-1">{t('printers.locationHelp')}</p>
  5660. </div>
  5661. <div className="flex items-center gap-2">
  5662. <input
  5663. type="checkbox"
  5664. id="auto_archive"
  5665. checked={form.auto_archive}
  5666. onChange={(e) => setForm({ ...form, auto_archive: e.target.checked })}
  5667. className="rounded border-bambu-dark-tertiary bg-bambu-dark text-bambu-green focus:ring-bambu-green"
  5668. />
  5669. <label htmlFor="auto_archive" className="text-sm text-bambu-gray">
  5670. {t('printers.modal.autoArchiveLabel')}
  5671. </label>
  5672. </div>
  5673. <button
  5674. type="button"
  5675. onClick={() => setShowDiagnostic(true)}
  5676. disabled={!form.ip_address.trim()}
  5677. className="w-full flex items-center justify-center gap-2 px-3 py-2 text-sm text-bambu-gray hover:text-white disabled:opacity-40 disabled:cursor-not-allowed border border-bambu-dark-tertiary rounded-lg transition-colors"
  5678. >
  5679. <Stethoscope className="w-4 h-4" />
  5680. {t('diagnostic.runButton')}
  5681. </button>
  5682. {saveWarning ? (
  5683. <div className="rounded-lg bg-amber-500/10 border border-amber-500/30 p-3 space-y-3">
  5684. <div className="flex items-start gap-2">
  5685. <AlertTriangle className="w-4 h-4 mt-0.5 flex-shrink-0 text-amber-400" />
  5686. <p className="text-sm text-amber-300">{t('printers.addPreflight.warning')}</p>
  5687. </div>
  5688. <DiagnosticChecklist result={saveWarning} />
  5689. <div className="flex gap-3">
  5690. <Button
  5691. type="button"
  5692. variant="secondary"
  5693. onClick={() => setSaveWarning(null)}
  5694. className="flex-1"
  5695. >
  5696. {t('printers.addPreflight.back')}
  5697. </Button>
  5698. <Button type="button" onClick={() => onAdd(form)} className="flex-1">
  5699. {t('printers.addPreflight.saveAnyway')}
  5700. </Button>
  5701. </div>
  5702. </div>
  5703. ) : (
  5704. <div className="flex gap-3 pt-2">
  5705. <Button type="button" variant="secondary" onClick={onClose} className="flex-1">
  5706. {t('common.cancel')}
  5707. </Button>
  5708. <Button type="submit" disabled={checkingSave} className="flex-1">
  5709. {checkingSave ? t('printers.addPreflight.checking') : t('printers.addPrinter')}
  5710. </Button>
  5711. </div>
  5712. )}
  5713. </form>
  5714. </CardContent>
  5715. </Card>
  5716. </div>
  5717. {showDiagnostic && (
  5718. <ConnectionDiagnosticModal
  5719. connection={{
  5720. ip_address: form.ip_address.trim(),
  5721. serial_number: form.serial_number.trim() || undefined,
  5722. access_code: form.access_code || undefined,
  5723. }}
  5724. printerName={form.name || null}
  5725. onClose={() => setShowDiagnostic(false)}
  5726. />
  5727. )}
  5728. </>
  5729. );
  5730. }
  5731. function FirmwareUpdateModal({
  5732. printer,
  5733. firmwareInfo,
  5734. onClose,
  5735. }: {
  5736. printer: Printer;
  5737. firmwareInfo: FirmwareUpdateInfo;
  5738. onClose: () => void;
  5739. }) {
  5740. const { t } = useTranslation();
  5741. const queryClient = useQueryClient();
  5742. const { showToast } = useToast();
  5743. const { hasPermission } = useAuth();
  5744. const canUpdate = hasPermission('firmware:update');
  5745. const [uploadStatus, setUploadStatus] = useState<FirmwareUploadStatus | null>(null);
  5746. const [isUploading, setIsUploading] = useState(false);
  5747. const [pollInterval, setPollInterval] = useState<NodeJS.Timeout | null>(null);
  5748. const [selectedVersion, setSelectedVersion] = useState<string | null>(
  5749. firmwareInfo.update_available ? firmwareInfo.latest_version : null,
  5750. );
  5751. // Prepare check query — runs when a version is selected and user can update
  5752. const { data: prepareInfo, isLoading: isPreparing } = useQuery({
  5753. queryKey: ['firmwarePrepare', printer.id, selectedVersion],
  5754. queryFn: () => firmwareApi.prepareUpload(printer.id, selectedVersion ?? undefined),
  5755. staleTime: 30000,
  5756. enabled: !!selectedVersion && canUpdate && !isUploading,
  5757. });
  5758. // Start upload mutation
  5759. const uploadMutation = useMutation({
  5760. mutationFn: () => firmwareApi.startUpload(printer.id, selectedVersion ?? undefined),
  5761. onSuccess: () => {
  5762. setIsUploading(true);
  5763. // Start polling for status
  5764. const interval = setInterval(async () => {
  5765. try {
  5766. const status = await firmwareApi.getUploadStatus(printer.id);
  5767. setUploadStatus(status);
  5768. if (status.status === 'complete' || status.status === 'error') {
  5769. clearInterval(interval);
  5770. setPollInterval(null);
  5771. setIsUploading(false);
  5772. if (status.status === 'complete') {
  5773. showToast(t('printers.firmwareModal.uploadedToast'), 'success');
  5774. queryClient.invalidateQueries({ queryKey: ['firmwareUpdate', printer.id] });
  5775. }
  5776. }
  5777. } catch {
  5778. // Ignore errors during polling
  5779. }
  5780. }, 2000);
  5781. setPollInterval(interval);
  5782. },
  5783. onError: (error: Error) => {
  5784. showToast(t('printers.firmwareModal.uploadFailed', { error: error.message }), 'error');
  5785. setIsUploading(false);
  5786. },
  5787. });
  5788. // Cleanup on unmount
  5789. useEffect(() => {
  5790. return () => {
  5791. if (pollInterval) clearInterval(pollInterval);
  5792. };
  5793. }, [pollInterval]);
  5794. const handleStartUpload = () => {
  5795. setUploadStatus(null);
  5796. uploadMutation.mutate();
  5797. };
  5798. return (
  5799. <div className="fixed inset-0 bg-black/50 flex items-center justify-center z-50">
  5800. <Card className="w-full max-w-md mx-4">
  5801. <CardContent>
  5802. <div className="flex items-start gap-3 mb-4">
  5803. <div className={`p-2 rounded-full ${firmwareInfo.update_available ? 'bg-orange-500/20' : 'bg-status-ok/20'}`}>
  5804. {firmwareInfo.update_available
  5805. ? <Download className="w-5 h-5 text-orange-400" />
  5806. : <CheckCircle className="w-5 h-5 text-status-ok" />}
  5807. </div>
  5808. <div className="flex-1">
  5809. <h3 className="text-lg font-semibold text-white">
  5810. {firmwareInfo.update_available ? t('printers.firmwareModal.title') : t('printers.firmwareModal.titleUpToDate')}
  5811. </h3>
  5812. <p className="text-sm text-bambu-gray mt-1">
  5813. {printer.name}
  5814. </p>
  5815. </div>
  5816. </div>
  5817. {/* Version Info */}
  5818. {(() => {
  5819. const selectedEntry = selectedVersion
  5820. ? firmwareInfo.available_versions?.find((v) => v.version === selectedVersion)
  5821. : null;
  5822. const displayVersion = selectedVersion ?? firmwareInfo.latest_version;
  5823. const displayNotes = selectedEntry?.release_notes ?? firmwareInfo.release_notes;
  5824. const showSecondLine = !!displayVersion && displayVersion !== firmwareInfo.current_version;
  5825. return (
  5826. <div className="bg-bambu-dark rounded-lg p-3 mb-4">
  5827. <div className="flex justify-between items-center text-sm">
  5828. <span className="text-bambu-gray">{t('printers.firmwareModal.currentVersion')}</span>
  5829. <span className={`font-mono ${showSecondLine ? 'text-white' : 'text-status-ok'}`}>
  5830. {firmwareInfo.current_version || t('common.unknown')}
  5831. </span>
  5832. </div>
  5833. {showSecondLine && (
  5834. <div className="flex justify-between items-center text-sm mt-1">
  5835. <span className="text-bambu-gray">{t('printers.firmwareModal.latestVersion')}</span>
  5836. <span className="text-orange-400 font-mono">{displayVersion}</span>
  5837. </div>
  5838. )}
  5839. {displayNotes && (
  5840. <details className="mt-3 text-sm" open={!showSecondLine} key={displayVersion ?? 'none'}>
  5841. <summary className={`cursor-pointer hover:underline ${showSecondLine ? 'text-orange-400' : 'text-status-ok'}`}>
  5842. {t('printers.firmwareModal.releaseNotes')}
  5843. </summary>
  5844. <div className="mt-2 text-bambu-gray text-xs max-h-40 overflow-y-auto whitespace-pre-wrap">
  5845. {displayNotes}
  5846. </div>
  5847. </details>
  5848. )}
  5849. </div>
  5850. );
  5851. })()}
  5852. {/* Available versions list */}
  5853. {firmwareInfo.available_versions && firmwareInfo.available_versions.length > 0 && !isUploading && uploadStatus?.status !== 'complete' && (
  5854. <div className="mb-4">
  5855. <div className="text-xs text-bambu-gray mb-2">{t('printers.firmwareModal.availableVersions')}</div>
  5856. <div className="max-h-56 overflow-y-auto border border-bambu-dark-tertiary rounded-lg divide-y divide-bambu-dark-tertiary">
  5857. {firmwareInfo.available_versions.map((v) => {
  5858. const isCurrent = firmwareInfo.current_version === v.version;
  5859. const isSelected = selectedVersion === v.version;
  5860. const cmp = firmwareInfo.current_version
  5861. ? compareFwVersions(v.version, firmwareInfo.current_version)
  5862. : 0;
  5863. const relLabel = isCurrent
  5864. ? t('printers.firmwareModal.currentBadge')
  5865. : cmp > 0
  5866. ? t('printers.firmwareModal.newerBadge')
  5867. : t('printers.firmwareModal.olderBadge');
  5868. const relClass = isCurrent
  5869. ? 'text-bambu-gray'
  5870. : cmp > 0
  5871. ? 'text-orange-400'
  5872. : 'text-blue-400';
  5873. return (
  5874. <button
  5875. key={v.version}
  5876. type="button"
  5877. disabled={!v.file_available || !canUpdate || isCurrent}
  5878. onClick={() => setSelectedVersion(v.version)}
  5879. className={`w-full text-left px-3 py-2 text-sm flex items-center justify-between gap-2 transition-colors ${
  5880. isSelected ? 'bg-orange-500/10' : 'hover:bg-bambu-dark'
  5881. } ${!v.file_available || !canUpdate || isCurrent ? 'opacity-60 cursor-not-allowed' : 'cursor-pointer'}`}
  5882. >
  5883. <div className="flex items-center gap-2 min-w-0">
  5884. <span className="font-mono text-white">{v.version}</span>
  5885. <span className={`text-xs ${relClass}`}>{relLabel}</span>
  5886. </div>
  5887. <span className={`text-xs px-2 py-0.5 rounded-full ${
  5888. isCurrent
  5889. ? 'bg-blue-500/15 text-blue-400 border border-blue-500/30'
  5890. : v.file_available
  5891. ? 'bg-bambu-green/15 text-bambu-green border border-bambu-green/30'
  5892. : 'bg-bambu-gray/10 text-bambu-gray border border-bambu-gray/30'
  5893. }`}>
  5894. {isCurrent
  5895. ? t('printers.firmwareModal.installed')
  5896. : v.file_available
  5897. ? t('printers.firmwareModal.usable')
  5898. : t('printers.firmwareModal.unavailable')}
  5899. </span>
  5900. </button>
  5901. );
  5902. })}
  5903. </div>
  5904. </div>
  5905. )}
  5906. {/* Status / Progress (only when a version is selected) */}
  5907. {!selectedVersion ? null : isPreparing ? (
  5908. <div className="flex items-center gap-2 text-bambu-gray text-sm mb-4">
  5909. <Loader2 className="w-4 h-4 animate-spin" />
  5910. {t('printers.firmwareModal.checkingPrereqs')}
  5911. </div>
  5912. ) : prepareInfo && !isUploading && !uploadStatus ? (
  5913. <div className="mb-4">
  5914. {prepareInfo.can_proceed ? (
  5915. <div className="flex items-center gap-2 text-bambu-green text-sm">
  5916. <Box className="w-4 h-4" />
  5917. {t('printers.firmwareModal.sdCardReady')}
  5918. </div>
  5919. ) : (
  5920. <div className="space-y-1">
  5921. {prepareInfo.errors.map((error, i) => (
  5922. <div key={i} className="flex items-center gap-2 text-red-400 text-sm">
  5923. <AlertCircle className="w-4 h-4 flex-shrink-0" />
  5924. {error}
  5925. </div>
  5926. ))}
  5927. </div>
  5928. )}
  5929. </div>
  5930. ) : null}
  5931. {/* Upload Progress */}
  5932. {(isUploading || uploadStatus) && uploadStatus && (
  5933. <div className="mb-4">
  5934. <div className="flex items-center justify-between text-sm mb-1">
  5935. <span className="text-bambu-gray capitalize">{uploadStatus.status}</span>
  5936. <span className="text-white">{uploadStatus.progress}%</span>
  5937. </div>
  5938. <div className="w-full bg-bambu-dark-tertiary rounded-full h-2">
  5939. <div
  5940. className={`h-2 rounded-full transition-all ${
  5941. uploadStatus.status === 'error' ? 'bg-status-error' :
  5942. uploadStatus.status === 'complete' ? 'bg-status-ok' : 'bg-orange-500'
  5943. } ${uploadStatus.status === 'uploading' ? 'animate-pulse' : ''}`}
  5944. style={{ width: `${uploadStatus.progress}%` }}
  5945. />
  5946. </div>
  5947. <p className="text-xs text-bambu-gray mt-1">{uploadStatus.message}</p>
  5948. {uploadStatus.error && (
  5949. <p className="text-xs text-red-400 mt-1">{uploadStatus.error}</p>
  5950. )}
  5951. </div>
  5952. )}
  5953. {/* Success Message */}
  5954. {uploadStatus?.status === 'complete' && (
  5955. <div className="bg-bambu-green/10 border border-bambu-green/30 rounded-lg p-3 mb-4">
  5956. <p className="text-sm text-bambu-green font-medium mb-2">
  5957. {t('printers.firmwareModal.uploadedSuccess')}
  5958. </p>
  5959. <p className="text-xs text-bambu-gray">
  5960. {t('printers.firmwareModal.applyInstructions')}
  5961. </p>
  5962. <ol className="text-xs text-bambu-gray mt-1 list-decimal list-inside space-y-1">
  5963. <li dangerouslySetInnerHTML={{ __html: t('printers.firmwareModal.step1') }} />
  5964. <li dangerouslySetInnerHTML={{ __html: t('printers.firmwareModal.step2') }} />
  5965. <li dangerouslySetInnerHTML={{ __html: t('printers.firmwareModal.step3') }} />
  5966. <li>{t('printers.firmwareModal.step4')}</li>
  5967. </ol>
  5968. </div>
  5969. )}
  5970. {/* Buttons */}
  5971. <div className="flex gap-2 justify-end">
  5972. <Button variant="secondary" onClick={onClose}>
  5973. {uploadStatus?.status === 'complete' ? t('printers.firmwareModal.done') : t('common.cancel')}
  5974. </Button>
  5975. {prepareInfo?.can_proceed && !isUploading && uploadStatus?.status !== 'complete' && canUpdate && (
  5976. <Button
  5977. onClick={handleStartUpload}
  5978. disabled={uploadMutation.isPending}
  5979. >
  5980. {uploadMutation.isPending ? (
  5981. <>
  5982. <Loader2 className="w-4 h-4 animate-spin mr-2" />
  5983. {t('printers.firmwareModal.starting')}
  5984. </>
  5985. ) : (
  5986. <>
  5987. <Download className="w-4 h-4 mr-2" />
  5988. {t('printers.firmwareModal.uploadFirmware')}
  5989. </>
  5990. )}
  5991. </Button>
  5992. )}
  5993. </div>
  5994. </CardContent>
  5995. </Card>
  5996. </div>
  5997. );
  5998. }
  5999. function EditPrinterModal({
  6000. printer,
  6001. onClose,
  6002. }: {
  6003. printer: Printer;
  6004. onClose: () => void;
  6005. }) {
  6006. const { t } = useTranslation();
  6007. const queryClient = useQueryClient();
  6008. const { showToast } = useToast();
  6009. const [form, setForm] = useState({
  6010. name: printer.name,
  6011. ip_address: printer.ip_address,
  6012. access_code: '',
  6013. model: printer.model || '',
  6014. location: printer.location || '',
  6015. auto_archive: printer.auto_archive,
  6016. });
  6017. // Setup-time pre-flight — same warn-on-save as the Add-Printer dialog, so an
  6018. // edit that breaks connectivity (e.g. a mistyped IP) is caught before save.
  6019. const [checkingSave, setCheckingSave] = useState(false);
  6020. const [saveWarning, setSaveWarning] = useState<PrinterDiagnosticResult | null>(null);
  6021. const updateMutation = useMutation({
  6022. mutationFn: (data: Partial<PrinterCreate>) => api.updatePrinter(printer.id, data),
  6023. onSuccess: () => {
  6024. queryClient.invalidateQueries({ queryKey: ['printers'] });
  6025. queryClient.invalidateQueries({ queryKey: ['printerStatus', printer.id] });
  6026. onClose();
  6027. },
  6028. onError: (error: Error) => showToast(error.message || t('printers.toast.failedToUpdate'), 'error'),
  6029. });
  6030. // Close on Escape key
  6031. useEffect(() => {
  6032. const handleKeyDown = (e: KeyboardEvent) => {
  6033. if (e.key === 'Escape') onClose();
  6034. };
  6035. window.addEventListener('keydown', handleKeyDown);
  6036. return () => window.removeEventListener('keydown', handleKeyDown);
  6037. }, [onClose]);
  6038. const doSave = () => {
  6039. const data: Partial<PrinterCreate> = {
  6040. name: form.name,
  6041. ip_address: form.ip_address,
  6042. model: form.model || undefined,
  6043. location: form.location || undefined,
  6044. auto_archive: form.auto_archive,
  6045. };
  6046. // Only include access_code if it was changed
  6047. if (form.access_code) {
  6048. data.access_code = form.access_code;
  6049. }
  6050. updateMutation.mutate(data);
  6051. };
  6052. const handleSubmit = async (e: React.FormEvent) => {
  6053. e.preventDefault();
  6054. setCheckingSave(true);
  6055. try {
  6056. const result = await api.diagnoseConnection({
  6057. ip_address: form.ip_address.trim(),
  6058. serial_number: printer.serial_number,
  6059. access_code: form.access_code || undefined,
  6060. });
  6061. if (result.checks.some((c) => c.status === 'fail')) {
  6062. setSaveWarning(result);
  6063. return;
  6064. }
  6065. } catch {
  6066. // Diagnostic infrastructure failed — never block the save on it.
  6067. } finally {
  6068. setCheckingSave(false);
  6069. }
  6070. doSave();
  6071. };
  6072. return (
  6073. <div
  6074. className="fixed inset-0 bg-black/50 flex items-start sm:items-center justify-center z-50 p-4 overflow-y-auto"
  6075. onClick={onClose}
  6076. >
  6077. <Card className="w-full max-w-md my-auto max-h-[calc(100vh-2rem)] overflow-y-auto" onClick={(e: React.MouseEvent) => e.stopPropagation()}>
  6078. <CardContent>
  6079. <h2 className="text-xl font-semibold mb-4">{t('printers.editPrinter')}</h2>
  6080. <form onSubmit={handleSubmit} className="space-y-4">
  6081. <div>
  6082. <label className="block text-sm text-bambu-gray mb-1">{t('printers.name')}</label>
  6083. <input
  6084. type="text"
  6085. required
  6086. 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"
  6087. value={form.name}
  6088. onChange={(e) => setForm({ ...form, name: e.target.value })}
  6089. placeholder={t('printers.modal.myPrinter')}
  6090. />
  6091. </div>
  6092. <div>
  6093. <label className="block text-sm text-bambu-gray mb-1">{t('printers.ipAddress')}</label>
  6094. <input
  6095. type="text"
  6096. required
  6097. pattern="(\d{1,3}(\.\d{1,3}){3}|[a-zA-Z0-9]([a-zA-Z0-9\-]{0,61}[a-zA-Z0-9])?(\.[a-zA-Z0-9]([a-zA-Z0-9\-]{0,61}[a-zA-Z0-9])?)*)"
  6098. 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"
  6099. value={form.ip_address}
  6100. onChange={(e) => setForm({ ...form, ip_address: e.target.value })}
  6101. placeholder="192.168.1.100 or printer.local"
  6102. />
  6103. </div>
  6104. <div>
  6105. <label className="block text-sm text-bambu-gray mb-1">{t('printers.serialNumber')}</label>
  6106. <input
  6107. type="text"
  6108. disabled
  6109. className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-bambu-gray cursor-not-allowed"
  6110. value={printer.serial_number}
  6111. />
  6112. <p className="text-xs text-bambu-gray mt-1">{t('printers.serialCannotBeChanged')}</p>
  6113. </div>
  6114. <div>
  6115. <label className="block text-sm text-bambu-gray mb-1">{t('printers.accessCode')}</label>
  6116. <input
  6117. type="password"
  6118. 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"
  6119. value={form.access_code}
  6120. onChange={(e) => setForm({ ...form, access_code: e.target.value })}
  6121. placeholder={t('printers.accessCodePlaceholder')}
  6122. />
  6123. </div>
  6124. <div>
  6125. <label className="block text-sm text-bambu-gray mb-1">{t('printers.model')}</label>
  6126. <select
  6127. 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"
  6128. value={form.model}
  6129. onChange={(e) => setForm({ ...form, model: e.target.value })}
  6130. >
  6131. <option value="">{t('printers.modal.selectModel')}</option>
  6132. <optgroup label="H2 Series">
  6133. <option value="H2C">H2C</option>
  6134. <option value="H2D">H2D</option>
  6135. <option value="H2D Pro">H2D Pro</option>
  6136. <option value="H2S">H2S</option>
  6137. </optgroup>
  6138. <optgroup label="X2 Series">
  6139. <option value="X2D">X2D</option>
  6140. </optgroup>
  6141. <optgroup label="X1 Series">
  6142. <option value="X1E">X1E</option>
  6143. <option value="X1C">X1 Carbon</option>
  6144. <option value="X1">X1</option>
  6145. </optgroup>
  6146. <optgroup label="P Series">
  6147. <option value="P2S">P2S</option>
  6148. <option value="P1S">P1S</option>
  6149. <option value="P1P">P1P</option>
  6150. </optgroup>
  6151. <optgroup label="A1 Series">
  6152. <option value="A1">A1</option>
  6153. <option value="A1 Mini">A1 Mini</option>
  6154. </optgroup>
  6155. </select>
  6156. </div>
  6157. <div>
  6158. <label className="block text-sm text-bambu-gray mb-1">Location / Group</label>
  6159. <input
  6160. type="text"
  6161. 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"
  6162. value={form.location}
  6163. onChange={(e) => setForm({ ...form, location: e.target.value })}
  6164. placeholder={t('printers.modal.locationPlaceholder')}
  6165. />
  6166. <p className="text-xs text-bambu-gray mt-1">{t('printers.locationHelp')}</p>
  6167. </div>
  6168. <div className="flex items-center gap-2">
  6169. <input
  6170. type="checkbox"
  6171. id="edit_auto_archive"
  6172. checked={form.auto_archive}
  6173. onChange={(e) => setForm({ ...form, auto_archive: e.target.checked })}
  6174. className="rounded border-bambu-dark-tertiary bg-bambu-dark text-bambu-green focus:ring-bambu-green"
  6175. />
  6176. <label htmlFor="edit_auto_archive" className="text-sm text-bambu-gray">
  6177. {t('printers.modal.autoArchiveLabel')}
  6178. </label>
  6179. </div>
  6180. {saveWarning ? (
  6181. <div className="rounded-lg bg-amber-500/10 border border-amber-500/30 p-3 space-y-3">
  6182. <div className="flex items-start gap-2">
  6183. <AlertTriangle className="w-4 h-4 mt-0.5 flex-shrink-0 text-amber-400" />
  6184. <p className="text-sm text-amber-300">{t('printers.addPreflight.warning')}</p>
  6185. </div>
  6186. <DiagnosticChecklist result={saveWarning} />
  6187. <div className="flex gap-3">
  6188. <Button
  6189. type="button"
  6190. variant="secondary"
  6191. onClick={() => setSaveWarning(null)}
  6192. className="flex-1"
  6193. >
  6194. {t('printers.addPreflight.back')}
  6195. </Button>
  6196. <Button
  6197. type="button"
  6198. onClick={doSave}
  6199. className="flex-1"
  6200. disabled={updateMutation.isPending}
  6201. >
  6202. {t('printers.addPreflight.saveAnyway')}
  6203. </Button>
  6204. </div>
  6205. </div>
  6206. ) : (
  6207. <div className="flex gap-3 pt-4">
  6208. <Button type="button" variant="secondary" onClick={onClose} className="flex-1">
  6209. {t('common.cancel')}
  6210. </Button>
  6211. <Button
  6212. type="submit"
  6213. className="flex-1"
  6214. disabled={updateMutation.isPending || checkingSave}
  6215. >
  6216. {checkingSave
  6217. ? t('printers.addPreflight.checking')
  6218. : updateMutation.isPending
  6219. ? t('common.saving')
  6220. : t('printers.modal.saveChanges')}
  6221. </Button>
  6222. </div>
  6223. )}
  6224. </form>
  6225. </CardContent>
  6226. </Card>
  6227. </div>
  6228. );
  6229. }
  6230. // Component to check if a printer is offline (for power dropdown)
  6231. function usePrinterOfflineStatus(printerId: number) {
  6232. const { data: status } = useQuery({
  6233. queryKey: ['printerStatus', printerId],
  6234. queryFn: () => api.getPrinterStatus(printerId),
  6235. refetchInterval: 30000,
  6236. });
  6237. return !status?.connected;
  6238. }
  6239. // Power dropdown item for an offline printer
  6240. function PowerDropdownItem({
  6241. printer,
  6242. plug,
  6243. onPowerOn,
  6244. isPowering,
  6245. }: {
  6246. printer: Printer;
  6247. plug: { id: number; name: string };
  6248. onPowerOn: (plugId: number) => void;
  6249. isPowering: boolean;
  6250. }) {
  6251. const isOffline = usePrinterOfflineStatus(printer.id);
  6252. // Fetch plug status
  6253. const { data: plugStatus } = useQuery({
  6254. queryKey: ['smartPlugStatus', plug.id],
  6255. queryFn: () => api.getSmartPlugStatus(plug.id),
  6256. refetchInterval: 10000,
  6257. });
  6258. // Only show if printer is offline
  6259. if (!isOffline) {
  6260. return null;
  6261. }
  6262. return (
  6263. <div className="flex items-center justify-between px-3 py-2 hover:bg-gray-100 dark:hover:bg-bambu-dark-tertiary">
  6264. <div className="flex items-center gap-2 min-w-0">
  6265. <span className="text-sm text-gray-900 dark:text-white truncate">{printer.name}</span>
  6266. {plugStatus && (
  6267. <span
  6268. className={`text-xs px-1.5 py-0.5 rounded ${
  6269. plugStatus.state === 'ON'
  6270. ? 'bg-bambu-green/20 text-bambu-green'
  6271. : 'bg-red-500/20 text-red-400'
  6272. }`}
  6273. >
  6274. {plugStatus.state || '?'}
  6275. </span>
  6276. )}
  6277. </div>
  6278. <button
  6279. onClick={() => onPowerOn(plug.id)}
  6280. disabled={isPowering || plugStatus?.state === 'ON'}
  6281. className={`px-2 py-1 text-xs rounded transition-colors flex items-center gap-1 ${
  6282. plugStatus?.state === 'ON'
  6283. ? 'bg-bambu-green/20 text-bambu-green cursor-default'
  6284. : 'bg-bambu-green/20 text-bambu-green hover:bg-bambu-green hover:text-white'
  6285. }`}
  6286. >
  6287. <Power className="w-3 h-3" />
  6288. {isPowering ? '...' : 'On'}
  6289. </button>
  6290. </div>
  6291. );
  6292. }
  6293. export function PrintersPage() {
  6294. const { t } = useTranslation();
  6295. const [showAddModal, setShowAddModal] = useState(false);
  6296. const [hideDisconnected, setHideDisconnected] = useState(() => {
  6297. return localStorage.getItem('hideDisconnectedPrinters') === 'true';
  6298. });
  6299. const [showPowerDropdown, setShowPowerDropdown] = useState(false);
  6300. const [poweringOn, setPoweringOn] = useState<number | null>(null);
  6301. const [sortBy, setSortBy] = useState<SortOption>(() => {
  6302. return (localStorage.getItem('printerSortBy') as SortOption) || 'name';
  6303. });
  6304. const [sortAsc, setSortAsc] = useState<boolean>(() => {
  6305. return localStorage.getItem('printerSortAsc') !== 'false';
  6306. });
  6307. // Card size: 1=small, 2=medium, 3=large, 4=xl
  6308. const [cardSize, setCardSize] = useState<number>(() => {
  6309. const saved = localStorage.getItem('printerCardSize');
  6310. return saved ? parseInt(saved, 10) : 2; // Default to medium
  6311. });
  6312. // Derive viewMode from cardSize: S=compact, M/L/XL=expanded
  6313. const viewMode: ViewMode = cardSize === 1 ? 'compact' : 'expanded';
  6314. const [search, setSearch] = useState('');
  6315. const [statusFilter, setStatusFilter] = useState<string>('all');
  6316. const [locationFilter, setLocationFilter] = useState<string>('all');
  6317. const [statusCacheVersion, setStatusCacheVersion] = useState(0);
  6318. const [collapsedSections, setCollapsedSections] = useState<Record<string, boolean>>(() => {
  6319. try {
  6320. const saved = localStorage.getItem('printerCollapsedSections');
  6321. return saved ? JSON.parse(saved) : {};
  6322. } catch { return {}; }
  6323. });
  6324. const queryClient = useQueryClient();
  6325. const { showToast } = useToast();
  6326. const { hasPermission } = useAuth();
  6327. // Embedded camera viewer state - supports multiple simultaneous viewers
  6328. // Persisted to localStorage so cameras reopen after navigation
  6329. const [embeddedCameraPrinters, setEmbeddedCameraPrinters] = useState<Map<number, { id: number; name: string }>>(() => {
  6330. // Initialize from localStorage if camera_view_mode is embedded
  6331. const saved = localStorage.getItem('openEmbeddedCameras');
  6332. if (saved) {
  6333. try {
  6334. const cameras = JSON.parse(saved) as Array<{ id: number; name: string }>;
  6335. return new Map(cameras.map(c => [c.id, c]));
  6336. } catch {
  6337. return new Map();
  6338. }
  6339. }
  6340. return new Map();
  6341. });
  6342. // Persist open cameras to localStorage when they change
  6343. useEffect(() => {
  6344. const cameras = Array.from(embeddedCameraPrinters.values());
  6345. if (cameras.length > 0) {
  6346. localStorage.setItem('openEmbeddedCameras', JSON.stringify(cameras));
  6347. } else {
  6348. localStorage.removeItem('openEmbeddedCameras');
  6349. }
  6350. }, [embeddedCameraPrinters]);
  6351. const { data: printers, isLoading } = useQuery({
  6352. queryKey: ['printers'],
  6353. queryFn: api.getPrinters,
  6354. });
  6355. // Fetch the UI-rendering subset of settings. Uses /ui-preferences (not /settings)
  6356. // so users with printers:read but no settings:read still get the values needed
  6357. // to render the clear-plate button, drying presets, AMS thresholds, etc. (#1293).
  6358. const { data: settings } = useQuery({
  6359. queryKey: ['ui-preferences'],
  6360. queryFn: api.getUiPreferences,
  6361. });
  6362. // Compute drying presets: user-configured (from settings) merged over built-in defaults
  6363. const effectiveDryingPresets = useMemo(() => {
  6364. if (settings?.drying_presets) {
  6365. try {
  6366. const userPresets = JSON.parse(settings.drying_presets);
  6367. if (typeof userPresets === 'object' && userPresets !== null && Object.keys(userPresets).length > 0) {
  6368. return { ...DRYING_PRESETS, ...userPresets };
  6369. }
  6370. } catch { /* ignore parse errors, use defaults */ }
  6371. }
  6372. return DRYING_PRESETS;
  6373. }, [settings?.drying_presets]);
  6374. // Close embedded cameras if mode changes to 'window'
  6375. useEffect(() => {
  6376. if (settings?.camera_view_mode === 'window' && embeddedCameraPrinters.size > 0) {
  6377. setEmbeddedCameraPrinters(new Map());
  6378. }
  6379. }, [settings?.camera_view_mode, embeddedCameraPrinters.size]);
  6380. // Fetch all smart plugs to know which printers have them
  6381. const { data: smartPlugs } = useQuery({
  6382. queryKey: ['smart-plugs'],
  6383. queryFn: api.getSmartPlugs,
  6384. });
  6385. // Fetch maintenance overview for all printers to show badges
  6386. const { data: maintenanceOverview } = useQuery({
  6387. queryKey: ['maintenanceOverview'],
  6388. queryFn: api.getMaintenanceOverview,
  6389. staleTime: 60 * 1000, // 1 minute
  6390. });
  6391. // Fetch Spoolman status to enable link spool feature
  6392. const { data: spoolmanStatus } = useQuery({
  6393. queryKey: ['spoolman-status'],
  6394. queryFn: api.getSpoolmanStatus,
  6395. staleTime: 60 * 1000, // 1 minute
  6396. });
  6397. const spoolmanEnabled = spoolmanStatus?.enabled && spoolmanStatus?.connected;
  6398. // Fetch Spoolman settings to get sync mode
  6399. const { data: spoolmanSettings } = useQuery({
  6400. queryKey: ['spoolman-settings'],
  6401. queryFn: api.getSpoolmanSettings,
  6402. enabled: !!spoolmanEnabled,
  6403. staleTime: 60 * 1000, // 1 minute
  6404. });
  6405. const spoolmanSyncMode = spoolmanSettings?.spoolman_sync_mode;
  6406. // Fetch unlinked spools to know if link button should be enabled
  6407. const { data: unlinkedSpools } = useQuery({
  6408. queryKey: ['unlinked-spools'],
  6409. queryFn: api.getUnlinkedSpools,
  6410. enabled: !!spoolmanEnabled,
  6411. staleTime: 30 * 1000, // 30 seconds
  6412. });
  6413. const hasUnlinkedSpools = unlinkedSpools && unlinkedSpools.length > 0;
  6414. // Fetch linked spools map (tag -> spool_id) to know which spools are already in Spoolman
  6415. const { data: linkedSpoolsData } = useQuery({
  6416. queryKey: ['linked-spools'],
  6417. queryFn: api.getLinkedSpools,
  6418. enabled: !!spoolmanEnabled,
  6419. staleTime: 30 * 1000, // 30 seconds
  6420. });
  6421. const linkedSpools = linkedSpoolsData?.linked;
  6422. // Fetch spool assignments for inventory feature
  6423. const { data: spoolAssignments } = useQuery({
  6424. queryKey: ['spool-assignments'],
  6425. queryFn: () => api.getAssignments(),
  6426. enabled: hasPermission('inventory:view_assignments'),
  6427. staleTime: 30 * 1000,
  6428. });
  6429. const unassignMutation = useMutation({
  6430. mutationFn: ({ printerId, amsId, trayId }: { printerId: number; amsId: number; trayId: number }) =>
  6431. api.unassignSpool(printerId, amsId, trayId),
  6432. onSuccess: () => {
  6433. queryClient.invalidateQueries({ queryKey: ['spool-assignments'] });
  6434. },
  6435. });
  6436. const { data: spoolmanSpools, isLoading: spoolmanSpoolsLoading } = useQuery({
  6437. queryKey: ['spoolman-inventory-spools'],
  6438. queryFn: () => api.getSpoolmanInventorySpools(false),
  6439. enabled: !!spoolmanEnabled,
  6440. staleTime: 30 * 1000,
  6441. });
  6442. const { data: spoolmanSlotAssignments, isLoading: spoolmanAssignmentsLoading } = useQuery({
  6443. queryKey: ['spoolman-slot-assignments'],
  6444. queryFn: () => api.getSpoolmanSlotAssignments(),
  6445. enabled: !!spoolmanEnabled,
  6446. staleTime: 30 * 1000,
  6447. });
  6448. const unassignSpoolmanMutation = useMutation({
  6449. mutationFn: (spoolmanSpoolId: number) => api.unassignSpoolmanSlot(spoolmanSpoolId),
  6450. onSuccess: () => {
  6451. queryClient.invalidateQueries({ queryKey: ['spoolman-inventory-spools'] });
  6452. queryClient.invalidateQueries({ queryKey: ['spoolman-slot-assignments'] });
  6453. },
  6454. });
  6455. // Helper to find assignment for a specific slot
  6456. const getAssignment = (printerId: number, amsId: number | string, trayId: number | string): SpoolAssignment | undefined => {
  6457. return spoolAssignments?.find(
  6458. (a) => a.printer_id === printerId && a.ams_id === Number(amsId) && a.tray_id === Number(trayId)
  6459. );
  6460. };
  6461. // Create a map of printer_id -> maintenance info for quick lookup
  6462. const maintenanceByPrinter = maintenanceOverview?.reduce(
  6463. (acc, overview) => {
  6464. acc[overview.printer_id] = {
  6465. due_count: overview.due_count,
  6466. warning_count: overview.warning_count,
  6467. total_print_hours: overview.total_print_hours,
  6468. };
  6469. return acc;
  6470. },
  6471. {} as Record<number, PrinterMaintenanceInfo>
  6472. ) || {};
  6473. // Create a map of printer_id -> smart plug
  6474. const smartPlugByPrinter = smartPlugs?.reduce(
  6475. (acc, plug) => {
  6476. if (plug.printer_id) {
  6477. acc[plug.printer_id] = plug;
  6478. }
  6479. return acc;
  6480. },
  6481. {} as Record<number, typeof smartPlugs[0]>
  6482. ) || {};
  6483. const addMutation = useMutation({
  6484. mutationFn: api.createPrinter,
  6485. onSuccess: () => {
  6486. queryClient.invalidateQueries({ queryKey: ['printers'] });
  6487. queryClient.invalidateQueries({ queryKey: ['maintenanceOverview'] });
  6488. setShowAddModal(false);
  6489. },
  6490. onError: (error: Error) => {
  6491. // Localized message when the backend returns a stable error code;
  6492. // the raw message is an English fallback for non-UI clients.
  6493. if (error instanceof ApiError && error.code === 'printer_connection_failed') {
  6494. showToast(t('printers.toast.connectionFailedNotAdded'), 'error');
  6495. return;
  6496. }
  6497. showToast(error.message || t('printers.toast.failedToAdd'), 'error');
  6498. },
  6499. });
  6500. const powerOnMutation = useMutation({
  6501. mutationFn: (plugId: number) => api.controlSmartPlug(plugId, 'on'),
  6502. onSuccess: () => {
  6503. queryClient.invalidateQueries({ queryKey: ['smart-plugs'] });
  6504. setPoweringOn(null);
  6505. },
  6506. onError: () => {
  6507. setPoweringOn(null);
  6508. },
  6509. });
  6510. // Bulk selection state
  6511. const [selectedPrinterIds, setSelectedPrinterIds] = useState<Set<number>>(new Set());
  6512. const [isSelectionMode, setIsSelectionMode] = useState(false);
  6513. const [bulkConfirmAction, setBulkConfirmAction] = useState<'stop' | 'pause' | 'clearPlate' | null>(null);
  6514. const [bulkActionPending, setBulkActionPending] = useState(false);
  6515. const selectionMode = isSelectionMode || selectedPrinterIds.size > 0;
  6516. const toggleSelect = useCallback((id: number) => {
  6517. setSelectedPrinterIds(prev => {
  6518. const next = new Set(prev);
  6519. if (next.has(id)) next.delete(id);
  6520. else next.add(id);
  6521. return next;
  6522. });
  6523. }, []);
  6524. const clearSelection = useCallback(() => {
  6525. setSelectedPrinterIds(new Set());
  6526. setIsSelectionMode(false);
  6527. }, []);
  6528. // Escape key exits selection mode
  6529. useEffect(() => {
  6530. const handleKeyDown = (e: KeyboardEvent) => {
  6531. if (e.key === 'Escape' && selectionMode) {
  6532. clearSelection();
  6533. }
  6534. };
  6535. window.addEventListener('keydown', handleKeyDown);
  6536. return () => window.removeEventListener('keydown', handleKeyDown);
  6537. }, [selectionMode, clearSelection]);
  6538. const executeBulkAction = useCallback(async (action: 'stop' | 'pause' | 'resume' | 'clearPlate' | 'clearHMS') => {
  6539. setBulkActionPending(true);
  6540. const ids = Array.from(selectedPrinterIds);
  6541. // Filter to only applicable printers based on cached state
  6542. const applicableIds = ids.filter(id => {
  6543. const status = queryClient.getQueryData<{ connected: boolean; state: string | null; hms_errors?: HMSError[] }>(['printerStatus', id]);
  6544. if (!status?.connected) return false;
  6545. switch (action) {
  6546. case 'stop': return status.state === 'RUNNING' || status.state === 'PAUSE';
  6547. case 'pause': return status.state === 'RUNNING';
  6548. case 'resume': return status.state === 'PAUSE';
  6549. case 'clearPlate': return !!(status as { awaiting_plate_clear?: boolean }).awaiting_plate_clear;
  6550. case 'clearHMS': return status.hms_errors && filterKnownHMSErrors(status.hms_errors).length > 0;
  6551. default: return false;
  6552. }
  6553. });
  6554. if (applicableIds.length === 0) {
  6555. showToast(t('printers.bulk.noneApplicable'), 'error');
  6556. setBulkActionPending(false);
  6557. setBulkConfirmAction(null);
  6558. return;
  6559. }
  6560. const apiCall = {
  6561. stop: api.stopPrint,
  6562. pause: api.pausePrint,
  6563. resume: api.resumePrint,
  6564. clearPlate: api.clearPlate,
  6565. clearHMS: api.clearHMSErrors,
  6566. }[action];
  6567. const results = await Promise.allSettled(
  6568. applicableIds.map(id => apiCall(id))
  6569. );
  6570. const succeeded = results.filter(r => r.status === 'fulfilled').length;
  6571. const failed = results.filter(r => r.status === 'rejected').length;
  6572. if (failed === 0) {
  6573. showToast(t('printers.bulk.success', { action: t(`printers.bulk.actions.${action}`), count: succeeded }));
  6574. } else {
  6575. showToast(t('printers.bulk.partial', { succeeded, failed }), 'error');
  6576. }
  6577. // Invalidate status queries for affected printers
  6578. applicableIds.forEach(id => {
  6579. queryClient.invalidateQueries({ queryKey: ['printerStatus', id] });
  6580. });
  6581. setBulkActionPending(false);
  6582. setBulkConfirmAction(null);
  6583. }, [selectedPrinterIds, queryClient, showToast, t]);
  6584. const handleBulkAction = useCallback((action: 'stop' | 'pause' | 'resume' | 'clearPlate' | 'clearHMS') => {
  6585. // Actions that need confirmation
  6586. if (action === 'stop' || action === 'pause' || action === 'clearPlate') {
  6587. setBulkConfirmAction(action);
  6588. } else {
  6589. executeBulkAction(action);
  6590. }
  6591. }, [executeBulkAction]);
  6592. const toggleHideDisconnected = () => {
  6593. const newValue = !hideDisconnected;
  6594. setHideDisconnected(newValue);
  6595. localStorage.setItem('hideDisconnectedPrinters', String(newValue));
  6596. };
  6597. const handleSortChange = (newSort: SortOption) => {
  6598. setSortBy(newSort);
  6599. localStorage.setItem('printerSortBy', newSort);
  6600. };
  6601. const toggleSortDirection = () => {
  6602. const newAsc = !sortAsc;
  6603. setSortAsc(newAsc);
  6604. localStorage.setItem('printerSortAsc', String(newAsc));
  6605. };
  6606. // Grid classes based on card size (1=small, 2=medium, 3=large, 4=xl)
  6607. const getGridClasses = () => {
  6608. switch (cardSize) {
  6609. case 1: return 'grid-cols-1 sm:grid-cols-2 md:grid-cols-3 lg:grid-cols-4 xl:grid-cols-5'; // S: many small cards
  6610. case 2: return 'grid-cols-1 md:grid-cols-2 xl:grid-cols-3'; // M: medium cards
  6611. case 3: return 'grid-cols-1 lg:grid-cols-2'; // L: large cards, 2 columns max
  6612. case 4: return 'grid-cols-1'; // XL: single column, full width
  6613. default: return 'grid-cols-1 md:grid-cols-2 xl:grid-cols-3';
  6614. }
  6615. };
  6616. const cardSizeLabels = ['S', 'M', 'L', 'XL'];
  6617. // Increment version counter whenever a printer status cache entry is updated so
  6618. // filteredPrinters re-computes reactively on WebSocket-driven status changes.
  6619. useEffect(() => {
  6620. const unsubscribe = queryClient.getQueryCache().subscribe((event) => {
  6621. if (
  6622. event.type === 'updated' &&
  6623. Array.isArray(event.query.queryKey) &&
  6624. event.query.queryKey[0] === 'printerStatus'
  6625. ) {
  6626. setStatusCacheVersion(v => v + 1);
  6627. }
  6628. });
  6629. return unsubscribe;
  6630. }, [queryClient]);
  6631. // Filter printers by search term, status, and location
  6632. const filteredPrinters = useMemo(() => {
  6633. if (!printers) return [];
  6634. let result = printers;
  6635. // Text search
  6636. if (search.trim()) {
  6637. const q = search.trim().toLowerCase();
  6638. result = result.filter(p =>
  6639. p.name.toLowerCase().includes(q) ||
  6640. (p.model || '').toLowerCase().includes(q) ||
  6641. (p.location || '').toLowerCase().includes(q) ||
  6642. (p.serial_number || '').toLowerCase().includes(q)
  6643. );
  6644. }
  6645. // Location filter
  6646. if (locationFilter !== 'all') {
  6647. result = result.filter(p => (p.location || '') === locationFilter);
  6648. }
  6649. // Status filter
  6650. if (statusFilter !== 'all') {
  6651. result = result.filter(p => {
  6652. const status = queryClient.getQueryData<{ connected: boolean; state: string | null; hms_errors?: HMSError[] }>(['printerStatus', p.id]);
  6653. if (!status?.connected) return statusFilter === 'offline';
  6654. const hmsErrors = status.hms_errors ? filterKnownHMSErrors(status.hms_errors) : [];
  6655. switch (statusFilter) {
  6656. case 'printing': return status.state === 'RUNNING';
  6657. case 'paused': return status.state === 'PAUSE';
  6658. case 'finished': return status.state === 'FINISH';
  6659. case 'error': return status.state === 'FAILED' || hmsErrors.length > 0;
  6660. case 'idle': return status.state !== 'RUNNING' && status.state !== 'PAUSE' && status.state !== 'FINISH' && status.state !== 'FAILED' && hmsErrors.length === 0;
  6661. case 'offline': return false; // Connected printers are never offline
  6662. default: return true;
  6663. }
  6664. });
  6665. }
  6666. return result;
  6667. // eslint-disable-next-line react-hooks/exhaustive-deps -- statusCacheVersion is intentional: it forces recompute when WebSocket updates printer status cache
  6668. }, [printers, search, statusFilter, locationFilter, queryClient, statusCacheVersion]);
  6669. // Derive unique locations for the location filter dropdown
  6670. const availableLocations = useMemo(() => {
  6671. if (!printers) return [];
  6672. return [...new Set(printers.map(p => p.location || '').filter(Boolean))].sort();
  6673. }, [printers]);
  6674. // Sort printers based on selected option
  6675. const sortedPrinters = useMemo(() => {
  6676. const sorted = [...filteredPrinters];
  6677. switch (sortBy) {
  6678. case 'name':
  6679. sorted.sort((a, b) => a.name.localeCompare(b.name));
  6680. break;
  6681. case 'model':
  6682. sorted.sort((a, b) => (a.model || '').localeCompare(b.model || ''));
  6683. break;
  6684. case 'location':
  6685. // Sort by location, with ungrouped printers last
  6686. sorted.sort((a, b) => {
  6687. const locA = a.location || '';
  6688. const locB = b.location || '';
  6689. if (!locA && locB) return 1;
  6690. if (locA && !locB) return -1;
  6691. return locA.localeCompare(locB) || a.name.localeCompare(b.name);
  6692. });
  6693. break;
  6694. case 'status':
  6695. // Sort by status: HMS errors > printing > idle > offline
  6696. sorted.sort((a, b) => {
  6697. const statusA = queryClient.getQueryData<{ connected: boolean; state: string | null; hms_errors?: HMSError[] }>(['printerStatus', a.id]);
  6698. const statusB = queryClient.getQueryData<{ connected: boolean; state: string | null; hms_errors?: HMSError[] }>(['printerStatus', b.id]);
  6699. const getPriority = (s: typeof statusA) => {
  6700. if (!s?.connected) return 3; // offline
  6701. const hmsErrors = s.hms_errors ? filterKnownHMSErrors(s.hms_errors) : [];
  6702. if (hmsErrors.length > 0) return 0; // HMS errors - top priority
  6703. if (s.state === 'RUNNING') return 1; // printing
  6704. return 2; // idle
  6705. };
  6706. return getPriority(statusA) - getPriority(statusB);
  6707. });
  6708. break;
  6709. }
  6710. // Apply ascending/descending
  6711. if (!sortAsc) {
  6712. sorted.reverse();
  6713. }
  6714. return sorted;
  6715. }, [filteredPrinters, sortBy, sortAsc, queryClient]);
  6716. const selectAll = useCallback(() => {
  6717. setSelectedPrinterIds(new Set(sortedPrinters.map(p => p.id)));
  6718. setIsSelectionMode(true);
  6719. }, [sortedPrinters]);
  6720. const selectByState = useCallback((state: PrinterState) => {
  6721. setSelectedPrinterIds(prev => {
  6722. const next = new Set(prev);
  6723. sortedPrinters.forEach(p => {
  6724. const status = queryClient.getQueryData<{ connected: boolean; state: string | null; hms_errors?: HMSError[] }>(['printerStatus', p.id]);
  6725. if (classifyPrinterStatus(status) === state) next.add(p.id);
  6726. });
  6727. return next;
  6728. });
  6729. setIsSelectionMode(true);
  6730. }, [sortedPrinters, queryClient]);
  6731. const selectByLocation = useCallback((location: string) => {
  6732. setSelectedPrinterIds(prev => {
  6733. const next = new Set(prev);
  6734. sortedPrinters.filter(p => (p.location || '') === location).forEach(p => next.add(p.id));
  6735. return next;
  6736. });
  6737. setIsSelectionMode(true);
  6738. }, [sortedPrinters]);
  6739. const selectByModel = useCallback((model: string) => {
  6740. setSelectedPrinterIds(prev => {
  6741. const next = new Set(prev);
  6742. sortedPrinters.filter(p => (p.model || 'Unknown') === model).forEach(p => next.add(p.id));
  6743. return next;
  6744. });
  6745. setIsSelectionMode(true);
  6746. }, [sortedPrinters]);
  6747. const toggleSectionCollapse = useCallback((key: string) => {
  6748. setCollapsedSections(prev => {
  6749. const next = { ...prev, [key]: !prev[key] };
  6750. try { localStorage.setItem('printerCollapsedSections', JSON.stringify(next)); } catch { /* quota exceeded / private mode */ }
  6751. return next;
  6752. });
  6753. }, []);
  6754. // Group printers when sorted by location, status, or model
  6755. const groupedPrinters = useMemo(() => {
  6756. if (sortBy === 'name') return null;
  6757. const groups: Record<string, typeof sortedPrinters> = {};
  6758. if (sortBy === 'location') {
  6759. sortedPrinters.forEach(printer => {
  6760. const location = printer.location || 'Ungrouped';
  6761. if (!groups[location]) groups[location] = [];
  6762. groups[location].push(printer);
  6763. });
  6764. } else if (sortBy === 'model') {
  6765. sortedPrinters.forEach(printer => {
  6766. const model = printer.model || 'Unknown';
  6767. if (!groups[model]) groups[model] = [];
  6768. groups[model].push(printer);
  6769. });
  6770. } else if (sortBy === 'status') {
  6771. sortedPrinters.forEach(printer => {
  6772. const status = queryClient.getQueryData<{ connected: boolean; state: string | null; hms_errors?: HMSError[] }>(['printerStatus', printer.id]);
  6773. const group = classifyPrinterStatus(status);
  6774. if (!groups[group]) groups[group] = [];
  6775. groups[group].push(printer);
  6776. });
  6777. }
  6778. return groups;
  6779. // eslint-disable-next-line react-hooks/exhaustive-deps -- classifyPrinterStatus & filterKnownHMSErrors are stable module-level functions, not reactive deps; statusCacheVersion forces recompute on WebSocket status updates
  6780. }, [sortBy, sortedPrinters, queryClient, statusCacheVersion]);
  6781. const toolbarRef = useRef<HTMLDivElement>(null);
  6782. const expandedToolbarControlsRef = useRef<HTMLDivElement>(null);
  6783. const expandedToolbarWidthRef = useRef(0);
  6784. const [compactToolbar, setCompactToolbar] = useState(false);
  6785. const measureToolbar = useCallback(() => {
  6786. const toolbar = toolbarRef.current;
  6787. if (!toolbar) return;
  6788. const measuredControlsWidth = expandedToolbarControlsRef.current?.offsetWidth;
  6789. if (measuredControlsWidth) {
  6790. expandedToolbarWidthRef.current = measuredControlsWidth;
  6791. }
  6792. const searchMinimumWidth = 220;
  6793. const gapWidth = 8;
  6794. const shouldCompact = expandedToolbarWidthRef.current > 0 && toolbar.clientWidth < expandedToolbarWidthRef.current + searchMinimumWidth + gapWidth;
  6795. setCompactToolbar(prev => (prev === shouldCompact ? prev : shouldCompact));
  6796. }, []);
  6797. const smartPlugCount = Object.keys(smartPlugByPrinter).length;
  6798. useLayoutEffect(() => {
  6799. measureToolbar();
  6800. const toolbar = toolbarRef.current;
  6801. if (!toolbar) return;
  6802. if (typeof ResizeObserver === 'undefined') {
  6803. window.addEventListener('resize', measureToolbar);
  6804. return () => window.removeEventListener('resize', measureToolbar);
  6805. }
  6806. const resizeObserver = new ResizeObserver(() => measureToolbar());
  6807. resizeObserver.observe(toolbar);
  6808. window.addEventListener('resize', measureToolbar);
  6809. return () => {
  6810. resizeObserver.disconnect();
  6811. window.removeEventListener('resize', measureToolbar);
  6812. };
  6813. }, [
  6814. measureToolbar,
  6815. printers?.length,
  6816. availableLocations.length,
  6817. hideDisconnected,
  6818. smartPlugCount,
  6819. ]);
  6820. const renderFilterControls = (inMenu = false) => (
  6821. <>
  6822. {/* Status filter */}
  6823. {printers && printers.length > 0 && (
  6824. <ToolbarDropdown
  6825. value={statusFilter}
  6826. onChange={setStatusFilter}
  6827. fullWidth={inMenu}
  6828. options={[
  6829. { value: 'all', label: t('printers.filter.allStatuses') },
  6830. { value: 'printing', label: t('printers.status.printing') },
  6831. { value: 'paused', label: t('printers.status.paused') },
  6832. { value: 'idle', label: t('printers.status.idle') },
  6833. { value: 'finished', label: t('printers.status.finished') },
  6834. { value: 'error', label: t('printers.status.error') },
  6835. { value: 'offline', label: t('printers.status.offline') },
  6836. ]}
  6837. />
  6838. )}
  6839. {/* Location filter — only shown when at least one printer has a location */}
  6840. {printers && printers.length > 0 && availableLocations.length > 0 && (
  6841. <ToolbarDropdown
  6842. value={locationFilter}
  6843. onChange={setLocationFilter}
  6844. fullWidth={inMenu}
  6845. options={[
  6846. { value: 'all', label: t('printers.filter.allLocations') },
  6847. ...availableLocations.map(loc => ({ value: loc, label: loc })),
  6848. ]}
  6849. />
  6850. )}
  6851. <button
  6852. type="button"
  6853. onClick={toggleHideDisconnected}
  6854. aria-pressed={hideDisconnected}
  6855. className={`h-8 px-2 rounded-lg border text-sm font-medium transition-colors ${inMenu ? 'w-full' : ''} ${
  6856. hideDisconnected
  6857. ? 'bg-bambu-green border-bambu-green text-white'
  6858. : 'bg-bambu-dark border-bambu-dark-tertiary text-white hover:bg-bambu-dark-tertiary'
  6859. }`}
  6860. >
  6861. {t('printers.hideOffline')}
  6862. </button>
  6863. </>
  6864. );
  6865. const renderViewControls = (inMenu = false) => (
  6866. <>
  6867. {/* Sort dropdown */}
  6868. <div className={`flex items-center gap-1 ${inMenu ? 'w-full' : ''}`}>
  6869. <ToolbarDropdown<SortOption>
  6870. value={sortBy}
  6871. onChange={handleSortChange}
  6872. fullWidth={inMenu}
  6873. options={[
  6874. { value: 'name', label: t('printers.sort.name') },
  6875. { value: 'status', label: t('printers.sort.status') },
  6876. { value: 'model', label: t('printers.sort.model') },
  6877. { value: 'location', label: t('printers.sort.location') },
  6878. ]}
  6879. />
  6880. <button
  6881. onClick={toggleSortDirection}
  6882. className="h-8 shrink-0 px-2 rounded-lg border bg-bambu-dark border-bambu-dark-tertiary text-white hover:bg-bambu-dark-tertiary transition-colors flex items-center justify-center"
  6883. title={sortAsc ? t('printers.sort.descending') : t('printers.sort.ascending')}
  6884. >
  6885. {sortAsc ? (
  6886. <ArrowUp className="w-4 h-4 text-white" />
  6887. ) : (
  6888. <ArrowDown className="w-4 h-4 text-white" />
  6889. )}
  6890. </button>
  6891. </div>
  6892. {/* Card size selector */}
  6893. <div className={`flex h-8 items-center bg-bambu-dark rounded-lg border border-bambu-dark-tertiary ${inMenu ? 'w-full' : ''}`}>
  6894. {cardSizeLabels.map((label, index) => {
  6895. const size = index + 1;
  6896. const isSelected = cardSize === size;
  6897. return (
  6898. <button
  6899. key={label}
  6900. onClick={() => {
  6901. setCardSize(size);
  6902. localStorage.setItem('printerCardSize', String(size));
  6903. }}
  6904. className={`h-full px-2 text-xs font-medium transition-colors ${inMenu ? 'flex-1' : ''} ${
  6905. index === 0 ? 'rounded-l-lg' : ''
  6906. } ${
  6907. index === cardSizeLabels.length - 1 ? 'rounded-r-lg' : ''
  6908. } ${
  6909. isSelected
  6910. ? 'bg-bambu-green text-white'
  6911. : 'text-white hover:bg-bambu-dark-tertiary'
  6912. }`}
  6913. title={label === 'S' ? t('printers.cardSize.small') : label === 'M' ? t('printers.cardSize.medium') : label === 'L' ? t('printers.cardSize.large') : t('printers.cardSize.extraLarge')}
  6914. >
  6915. {label}
  6916. </button>
  6917. );
  6918. })}
  6919. </div>
  6920. </>
  6921. );
  6922. const renderActionControls = (inMenu = false) => (
  6923. <>
  6924. {/* Bulk select toggle */}
  6925. <button
  6926. onClick={() => {
  6927. if (selectionMode) clearSelection();
  6928. else setIsSelectionMode(true);
  6929. }}
  6930. className={`h-8 px-2 rounded-lg border transition-colors ${inMenu ? 'w-full justify-center gap-1.5 text-sm font-medium flex items-center' : ''} ${
  6931. selectionMode
  6932. ? 'bg-bambu-green border-bambu-green text-white'
  6933. : 'bg-bambu-dark border-bambu-dark-tertiary text-white hover:bg-bambu-dark-tertiary'
  6934. }`}
  6935. title={t('printers.bulk.select')}
  6936. disabled={!hasPermission('printers:control')}
  6937. >
  6938. <CheckSquare className="w-4 h-4" />
  6939. {inMenu && <span>{t('printers.bulk.select')}</span>}
  6940. </button>
  6941. {/* Power dropdown for offline printers with smart plugs */}
  6942. {hideDisconnected && Object.keys(smartPlugByPrinter).length > 0 && (
  6943. <div className={`relative ${inMenu ? 'w-full' : ''}`}>
  6944. <button
  6945. onClick={() => setShowPowerDropdown(!showPowerDropdown)}
  6946. className={`h-8 flex items-center gap-1.5 px-2 text-sm rounded-lg border transition-colors ${
  6947. inMenu
  6948. ? 'w-full justify-between bg-bambu-dark border-bambu-dark-tertiary text-white hover:bg-bambu-dark-tertiary hover:text-white'
  6949. : 'bg-bambu-dark border-bambu-dark-tertiary text-white hover:bg-bambu-dark-tertiary'
  6950. }`}
  6951. >
  6952. <span className="flex items-center gap-1.5">
  6953. <Power className="w-4 h-4" />
  6954. {t('printers.powerOn')}
  6955. </span>
  6956. <ChevronDown className={`w-3 h-3 transition-transform ${showPowerDropdown ? 'rotate-180' : ''}`} />
  6957. </button>
  6958. {showPowerDropdown && (
  6959. <>
  6960. {/* Backdrop to close dropdown */}
  6961. <div
  6962. className="fixed inset-0 z-10"
  6963. onClick={() => setShowPowerDropdown(false)}
  6964. />
  6965. <div className="absolute right-0 mt-2 w-56 bg-white dark:bg-bambu-dark-secondary border border-gray-200 dark:border-bambu-dark-tertiary rounded-lg shadow-lg z-20 py-1">
  6966. <div className="px-3 py-2 text-xs text-gray-500 dark:text-bambu-gray border-b border-gray-200 dark:border-bambu-dark-tertiary">
  6967. {t('printers.offlinePrintersWithPlugs')}
  6968. </div>
  6969. {printers?.filter(p => smartPlugByPrinter[p.id]).map(printer => (
  6970. <PowerDropdownItem
  6971. key={printer.id}
  6972. printer={printer}
  6973. plug={smartPlugByPrinter[printer.id]}
  6974. onPowerOn={(plugId) => {
  6975. setPoweringOn(plugId);
  6976. powerOnMutation.mutate(plugId);
  6977. }}
  6978. isPowering={poweringOn === smartPlugByPrinter[printer.id]?.id}
  6979. />
  6980. ))}
  6981. {printers?.filter(p => smartPlugByPrinter[p.id]).length === 0 && (
  6982. <div className="px-3 py-2 text-sm text-bambu-gray">
  6983. No printers with smart plugs
  6984. </div>
  6985. )}
  6986. </div>
  6987. </>
  6988. )}
  6989. </div>
  6990. )}
  6991. <Button
  6992. onClick={() => setShowAddModal(true)}
  6993. disabled={!hasPermission('printers:create')}
  6994. title={!hasPermission('printers:create') ? t('printers.permission.noAdd') : undefined}
  6995. className={`!h-8 !min-h-8 px-2 py-0 ${inMenu ? 'w-full' : ''}`}
  6996. >
  6997. <Plus className="w-4 h-4" />
  6998. {t('printers.addPrinter')}
  6999. </Button>
  7000. </>
  7001. );
  7002. return (
  7003. <div className="p-4 md:p-8">
  7004. <div className="space-y-3 mb-6">
  7005. <div>
  7006. <h1 className="text-2xl font-bold text-white flex items-center gap-3">
  7007. <PrinterIcon className="w-7 h-7 text-bambu-green" />
  7008. {t('printers.title')}
  7009. </h1>
  7010. <StatusSummaryBar printers={printers} />
  7011. </div>
  7012. <div ref={toolbarRef} className="relative flex items-center gap-2">
  7013. {/* Only show search bar when printers exist */}
  7014. {printers && printers.length > 0 && (
  7015. <div className="relative min-w-0 flex-1">
  7016. <Search className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray/50" />
  7017. <input
  7018. type="search"
  7019. name="printer-search"
  7020. autoComplete="off"
  7021. data-1p-ignore
  7022. data-lpignore="true"
  7023. value={search}
  7024. onChange={(e) => setSearch(e.target.value)}
  7025. placeholder={t('printers.search')}
  7026. aria-label={t('printers.search')}
  7027. className="w-full h-8 pl-9 pr-8 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white text-sm placeholder:text-bambu-gray/50 focus:outline-none focus:border-bambu-green"
  7028. />
  7029. {search && (
  7030. <button
  7031. type="button"
  7032. aria-label={t('common.clear')}
  7033. onClick={() => setSearch('')}
  7034. className="absolute right-3 top-1/2 -translate-y-1/2 text-bambu-gray hover:text-white"
  7035. >
  7036. <X className="w-4 h-4" />
  7037. </button>
  7038. )}
  7039. </div>
  7040. )}
  7041. <div
  7042. ref={expandedToolbarControlsRef}
  7043. aria-hidden={compactToolbar}
  7044. inert={compactToolbar}
  7045. className={`${compactToolbar ? 'absolute -left-[9999px] top-0 flex w-max pointer-events-none opacity-0' : 'flex'} ml-auto items-center justify-end gap-2 flex-nowrap [&>*]:shrink-0`}
  7046. >
  7047. <div className="h-6 w-px bg-bambu-dark-tertiary" />
  7048. <div className="flex items-center gap-2">{renderFilterControls()}</div>
  7049. <div className="h-6 w-px bg-bambu-dark-tertiary" />
  7050. <div className="flex items-center gap-2">{renderViewControls()}</div>
  7051. <div className="h-6 w-px bg-bambu-dark-tertiary" />
  7052. <div className="flex items-center gap-2">{renderActionControls()}</div>
  7053. </div>
  7054. {compactToolbar && (
  7055. <div className="ml-auto flex items-center justify-end gap-1">
  7056. <ToolbarMenu label={t('printers.toolbar.filters', 'Filters')} icon={<Filter className="w-4 h-4" />}>
  7057. <div className="flex w-48 flex-col gap-2">{renderFilterControls(true)}</div>
  7058. </ToolbarMenu>
  7059. <ToolbarMenu label={t('printers.toolbar.view', 'View')} icon={<SlidersHorizontal className="w-4 h-4" />}>
  7060. <div className="flex w-48 flex-col gap-2">{renderViewControls(true)}</div>
  7061. </ToolbarMenu>
  7062. <ToolbarMenu label={t('printers.toolbar.actions', 'Actions')} icon={<MoreHorizontal className="w-4 h-4" />}>
  7063. <div className="flex w-48 flex-col gap-2">{renderActionControls(true)}</div>
  7064. </ToolbarMenu>
  7065. </div>
  7066. )}
  7067. </div>
  7068. </div>
  7069. {isLoading ? (
  7070. <div className="text-center py-12 text-bambu-gray">{t('common.loading')}</div>
  7071. ) : printers?.length === 0 ? (
  7072. <Card>
  7073. <CardContent className="text-center py-12">
  7074. <p className="text-bambu-gray mb-4">{t('printers.noPrintersConfigured')}</p>
  7075. <Button
  7076. onClick={() => setShowAddModal(true)}
  7077. disabled={!hasPermission('printers:create')}
  7078. title={!hasPermission('printers:create') ? t('printers.permission.noAdd') : undefined}
  7079. >
  7080. <Plus className="w-4 h-4" />
  7081. {t('printers.addPrinter')}
  7082. </Button>
  7083. </CardContent>
  7084. </Card>
  7085. ) : sortedPrinters.length === 0 && (search.trim() || statusFilter !== 'all' || locationFilter !== 'all') ? (
  7086. <Card>
  7087. <CardContent className="text-center py-12">
  7088. <p className="text-bambu-gray">{t('printers.noSearchResults')}</p>
  7089. </CardContent>
  7090. </Card>
  7091. ) : groupedPrinters ? (
  7092. /* Grouped view (location, status, or model) */
  7093. <div className="space-y-6">
  7094. {(() => {
  7095. const keys = sortBy === 'status'
  7096. ? STATUS_GROUP_ORDER.filter(k => groupedPrinters[k]?.length > 0)
  7097. : Object.keys(groupedPrinters);
  7098. // For status grouping, asc/desc flips the fixed priority order
  7099. // (asc = error→offline, desc = offline→error). This matches the
  7100. // sort-toggle behaviour for other groupings.
  7101. return (sortAsc ? keys : [...keys].reverse());
  7102. })().map((groupKey) => {
  7103. const groupPrinters = groupedPrinters[groupKey];
  7104. const collapseKey = `${sortBy}:${groupKey}`;
  7105. const isOpen = !collapsedSections[collapseKey];
  7106. const dot = sortBy === 'status'
  7107. ? STATUS_GROUP_META[groupKey]?.dot || 'bg-bambu-green'
  7108. : 'bg-bambu-green';
  7109. const label = sortBy === 'status'
  7110. ? t(STATUS_GROUP_META[groupKey]?.labelKey || groupKey)
  7111. : groupKey;
  7112. return (
  7113. <Collapsible
  7114. key={groupKey}
  7115. open={isOpen}
  7116. onToggle={() => toggleSectionCollapse(collapseKey)}
  7117. summaryClassName="py-1"
  7118. summary={
  7119. <h2 className="text-lg font-semibold text-white flex items-center gap-2">
  7120. <span className={`w-2 h-2 rounded-full ${dot}`} />
  7121. {label}
  7122. <span className="text-sm font-normal text-bambu-gray">({groupPrinters.length})</span>
  7123. {selectionMode && (
  7124. <button
  7125. onClick={(e) => {
  7126. e.stopPropagation();
  7127. if (sortBy === 'location') selectByLocation(groupKey === 'Ungrouped' ? '' : groupKey);
  7128. else if (sortBy === 'status') selectByState(groupKey as PrinterState);
  7129. else if (sortBy === 'model') selectByModel(groupKey);
  7130. }}
  7131. className="text-xs text-bambu-green hover:text-bambu-green-light transition-colors ml-1"
  7132. >
  7133. {t('printers.bulk.selectAll')}
  7134. </button>
  7135. )}
  7136. </h2>
  7137. }
  7138. >
  7139. <div className={`grid gap-4 ${cardSize >= 3 ? 'gap-6' : ''} ${getGridClasses()}`}>
  7140. {groupPrinters.map((printer) => (
  7141. <PrinterCard
  7142. key={printer.id}
  7143. printer={printer}
  7144. hideIfDisconnected={hideDisconnected}
  7145. maintenanceInfo={maintenanceByPrinter[printer.id]}
  7146. viewMode={viewMode}
  7147. cardSize={cardSize}
  7148. amsThresholds={settings ? {
  7149. humidityGood: Number(settings.ams_humidity_good) || 40,
  7150. humidityFair: Number(settings.ams_humidity_fair) || 60,
  7151. tempGood: Number(settings.ams_temp_good) || 28,
  7152. tempFair: Number(settings.ams_temp_fair) || 35,
  7153. } : undefined}
  7154. spoolmanEnabled={spoolmanEnabled}
  7155. hasUnlinkedSpools={hasUnlinkedSpools}
  7156. linkedSpools={linkedSpools}
  7157. spoolmanUrl={spoolmanStatus?.url}
  7158. spoolmanSyncMode={spoolmanSyncMode}
  7159. onGetAssignment={getAssignment}
  7160. onUnassignSpool={(pid, aid, tid) => unassignMutation.mutate({ printerId: pid, amsId: aid, trayId: tid })}
  7161. spoolmanSpools={spoolmanSpools}
  7162. spoolmanSlotAssignments={spoolmanSlotAssignments}
  7163. spoolmanLoading={spoolmanSpoolsLoading || spoolmanAssignmentsLoading}
  7164. onUnassignSpoolmanSpool={(id) => unassignSpoolmanMutation.mutate(id)}
  7165. timeFormat={settings?.time_format || 'system'}
  7166. cameraViewMode={settings?.camera_view_mode || 'window'}
  7167. onOpenEmbeddedCamera={(id, name) => setEmbeddedCameraPrinters(prev => new Map(prev).set(id, { id, name }))}
  7168. checkPrinterFirmware={settings?.check_printer_firmware !== false}
  7169. dryingPresets={effectiveDryingPresets}
  7170. requirePlateClear={settings?.require_plate_clear === true}
  7171. selectionMode={selectionMode}
  7172. isSelected={selectedPrinterIds.has(printer.id)}
  7173. onToggleSelect={toggleSelect}
  7174. />
  7175. ))}
  7176. </div>
  7177. </Collapsible>
  7178. );
  7179. })}
  7180. </div>
  7181. ) : (
  7182. /* Regular grid view */
  7183. <div className={`grid gap-4 ${cardSize >= 3 ? 'gap-6' : ''} ${getGridClasses()}`}>
  7184. {sortedPrinters.map((printer) => (
  7185. <PrinterCard
  7186. key={printer.id}
  7187. printer={printer}
  7188. hideIfDisconnected={hideDisconnected}
  7189. maintenanceInfo={maintenanceByPrinter[printer.id]}
  7190. viewMode={viewMode}
  7191. cardSize={cardSize}
  7192. spoolmanEnabled={spoolmanEnabled}
  7193. hasUnlinkedSpools={hasUnlinkedSpools}
  7194. linkedSpools={linkedSpools}
  7195. spoolmanUrl={spoolmanStatus?.url}
  7196. spoolmanSyncMode={spoolmanSyncMode}
  7197. onGetAssignment={getAssignment}
  7198. onUnassignSpool={(pid, aid, tid) => unassignMutation.mutate({ printerId: pid, amsId: aid, trayId: tid })}
  7199. spoolmanSpools={spoolmanSpools}
  7200. spoolmanSlotAssignments={spoolmanSlotAssignments}
  7201. spoolmanLoading={spoolmanSpoolsLoading || spoolmanAssignmentsLoading}
  7202. onUnassignSpoolmanSpool={(id) => unassignSpoolmanMutation.mutate(id)}
  7203. amsThresholds={settings ? {
  7204. humidityGood: Number(settings.ams_humidity_good) || 40,
  7205. humidityFair: Number(settings.ams_humidity_fair) || 60,
  7206. tempGood: Number(settings.ams_temp_good) || 28,
  7207. tempFair: Number(settings.ams_temp_fair) || 35,
  7208. } : undefined}
  7209. timeFormat={settings?.time_format || 'system'}
  7210. cameraViewMode={settings?.camera_view_mode || 'window'}
  7211. onOpenEmbeddedCamera={(id, name) => setEmbeddedCameraPrinters(prev => new Map(prev).set(id, { id, name }))}
  7212. checkPrinterFirmware={settings?.check_printer_firmware !== false}
  7213. dryingPresets={effectiveDryingPresets}
  7214. requirePlateClear={settings?.require_plate_clear === true}
  7215. selectionMode={selectionMode}
  7216. isSelected={selectedPrinterIds.has(printer.id)}
  7217. onToggleSelect={toggleSelect}
  7218. />
  7219. ))}
  7220. </div>
  7221. )}
  7222. {showAddModal && (
  7223. <AddPrinterModal
  7224. onClose={() => setShowAddModal(false)}
  7225. onAdd={(data) => addMutation.mutate(data)}
  7226. existingSerials={printers?.map(p => p.serial_number) || []}
  7227. />
  7228. )}
  7229. {/* Bulk selection toolbar */}
  7230. {selectionMode && printers && (
  7231. <BulkPrinterToolbar
  7232. selectedIds={selectedPrinterIds}
  7233. printers={printers}
  7234. onClose={clearSelection}
  7235. onSelectAll={selectAll}
  7236. onSelectByLocation={selectByLocation}
  7237. onSelectByState={selectByState}
  7238. onAction={handleBulkAction}
  7239. actionPending={bulkActionPending}
  7240. />
  7241. )}
  7242. {/* Bulk action confirmation modals */}
  7243. {bulkConfirmAction === 'stop' && (
  7244. <ConfirmModal
  7245. title={t('printers.bulk.confirm.stopTitle', { count: selectedPrinterIds.size })}
  7246. message={t('printers.bulk.confirm.stopMessage', { count: selectedPrinterIds.size })}
  7247. confirmText={t('printers.bulk.confirm.stopButton')}
  7248. variant="danger"
  7249. isLoading={bulkActionPending}
  7250. onConfirm={() => executeBulkAction('stop')}
  7251. onCancel={() => setBulkConfirmAction(null)}
  7252. />
  7253. )}
  7254. {bulkConfirmAction === 'pause' && (
  7255. <ConfirmModal
  7256. title={t('printers.bulk.confirm.pauseTitle', { count: selectedPrinterIds.size })}
  7257. message={t('printers.bulk.confirm.pauseMessage', { count: selectedPrinterIds.size })}
  7258. confirmText={t('printers.bulk.confirm.pauseButton')}
  7259. isLoading={bulkActionPending}
  7260. onConfirm={() => executeBulkAction('pause')}
  7261. onCancel={() => setBulkConfirmAction(null)}
  7262. />
  7263. )}
  7264. {bulkConfirmAction === 'clearPlate' && (
  7265. <ConfirmModal
  7266. title={t('printers.bulk.confirm.clearPlateTitle', { count: selectedPrinterIds.size })}
  7267. message={t('printers.bulk.confirm.clearPlateMessage', { count: selectedPrinterIds.size })}
  7268. confirmText={t('printers.bulk.confirm.clearPlateButton')}
  7269. isLoading={bulkActionPending}
  7270. onConfirm={() => executeBulkAction('clearPlate')}
  7271. onCancel={() => setBulkConfirmAction(null)}
  7272. />
  7273. )}
  7274. {/* Embedded Camera Viewers - multiple viewers can be open simultaneously */}
  7275. {Array.from(embeddedCameraPrinters.values()).map((camera, index) => (
  7276. <EmbeddedCameraViewer
  7277. key={camera.id}
  7278. printerId={camera.id}
  7279. printerName={camera.name}
  7280. viewerIndex={index}
  7281. onClose={() => setEmbeddedCameraPrinters(prev => {
  7282. const next = new Map(prev);
  7283. next.delete(camera.id);
  7284. return next;
  7285. })}
  7286. />
  7287. ))}
  7288. </div>
  7289. );
  7290. }