PrintersPage.tsx 302 KB

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