| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436 |
- import { useState, useMemo, useEffect, useCallback, useRef } from 'react';
- import { useQuery, useMutation } from '@tanstack/react-query';
- import { useTranslation } from 'react-i18next';
- import { X, Loader2, Settings2, ChevronDown, CheckCircle2, RotateCcw } from 'lucide-react';
- import { api } from '../api/client';
- import type { KProfile } from '../api/client';
- import { Button } from './Button';
- interface SlotInfo {
- amsId: number;
- trayId: number;
- trayCount: number;
- trayType?: string;
- trayColor?: string;
- traySubBrands?: string;
- trayInfoIdx?: string;
- extruderId?: number;
- caliIdx?: number | null;
- savedPresetId?: string;
- }
- // Get proper AMS label (handles HT AMS with ID 128+)
- function getAmsLabel(amsId: number, trayCount: number): string {
- // External spool
- if (amsId === 255) return 'External';
- let normalizedId: number;
- let isHt = false;
- if (amsId >= 128 && amsId <= 135) {
- // HT AMS range: 128-135 → A-H
- normalizedId = amsId - 128;
- isHt = true;
- } else if (amsId >= 0 && amsId <= 3) {
- // Regular AMS range: 0-3 → A-D
- normalizedId = amsId;
- // Check tray count as secondary indicator
- isHt = trayCount === 1;
- } else {
- // Unknown range - fallback to A
- normalizedId = 0;
- }
- // Cap to valid letter range (A-H)
- normalizedId = Math.max(0, Math.min(normalizedId, 7));
- const letter = String.fromCharCode(65 + normalizedId);
- return isHt ? `HT-${letter}` : `AMS-${letter}`;
- }
- // Convert setting_id to tray_info_idx (filament_id format)
- // Bambu format: setting_id "GFSL05" → tray_info_idx "GFL05"
- function convertToTrayInfoIdx(settingId: string): string {
- // Strip version suffix if present (e.g., GFSL05_07 -> GFSL05)
- const baseId = settingId.includes('_') ? settingId.split('_')[0] : settingId;
- // Bambu presets start with "GFS" - remove the 'S' to get filament_id
- if (baseId.startsWith('GFS')) {
- return 'GF' + baseId.slice(3);
- }
- // User presets (PFUS*, PFSP*) - use the base setting_id (without version suffix)
- // This follows the pattern that filament_id and setting_id share the same base ID
- if (baseId.startsWith('PFUS') || baseId.startsWith('PFSP')) {
- return baseId; // Use base ID without version suffix
- }
- // For other formats, use as-is
- return baseId;
- }
- interface ConfigureAmsSlotModalProps {
- isOpen: boolean;
- onClose: () => void;
- printerId: number;
- slotInfo: SlotInfo;
- nozzleDiameter?: string;
- printerModel?: string;
- onSuccess?: () => void;
- fullScreen?: boolean;
- }
- // Known filament material types
- const MATERIAL_TYPES = ['PLA', 'PETG', 'PCTG', 'ABS', 'ASA', 'TPU', 'PC', 'PA', 'NYLON', 'PVA', 'HIPS', 'PP', 'PET'];
- // Extract filament type from preset name by finding known material type
- function parsePresetName(name: string): { material: string; brand: string; variant: string } {
- // Remove printer/nozzle suffix first
- const withoutSuffix = name.replace(/@.+$/, '').trim();
- const upperName = withoutSuffix.toUpperCase();
- // Handle "X Support for Y" pattern: the filament type is Y, not X.
- // e.g. "PLA Support for PETG PETG Basic" → material is PETG
- const supportMatch = upperName.match(/\bSUPPORT\s+FOR\s+/);
- if (supportMatch) {
- const afterSupport = upperName.slice(supportMatch.index! + supportMatch[0].length);
- for (const mat of MATERIAL_TYPES) {
- const regex = new RegExp(`\\b${mat}\\b`);
- if (regex.test(afterSupport)) {
- const brand = withoutSuffix.slice(0, supportMatch.index).trim();
- return { material: mat, brand, variant: 'Support' };
- }
- }
- }
- // Try to find a known material type in the name
- for (const mat of MATERIAL_TYPES) {
- // Use word boundary to match whole words only
- const regex = new RegExp(`\\b${mat}\\b`, 'i');
- if (regex.test(upperName)) {
- // Found material, extract brand (everything before material) and variant (after)
- const parts = withoutSuffix.split(regex);
- const brand = parts[0]?.trim() || '';
- const variant = parts[1]?.trim() || '';
- return { material: mat, brand, variant };
- }
- }
- // Fallback: assume first word is brand, second is material
- const parts = withoutSuffix.split(/\s+/);
- if (parts.length >= 2) {
- return { material: parts[1], brand: parts[0], variant: parts.slice(2).join(' ') };
- }
- return { material: withoutSuffix, brand: '', variant: '' };
- }
- // Check if a preset is a user preset (not built-in)
- function isUserPreset(settingId: string): boolean {
- // Built-in presets have specific patterns, user presets are UUIDs
- return !settingId.startsWith('GF') && !settingId.startsWith('P1');
- }
- // Common color name to hex mapping
- const COLOR_NAME_MAP: Record<string, string> = {
- // Basic colors
- 'white': 'FFFFFF',
- 'black': '000000',
- 'red': 'FF0000',
- 'green': '00FF00',
- 'blue': '0000FF',
- 'yellow': 'FFFF00',
- 'cyan': '00FFFF',
- 'magenta': 'FF00FF',
- 'orange': 'FFA500',
- 'purple': '800080',
- 'pink': 'FFC0CB',
- 'brown': '8B4513',
- 'gray': '808080',
- 'grey': '808080',
- // Filament-specific colors
- 'jade white': 'FFFEF2',
- 'ivory': 'FFFFF0',
- 'beige': 'F5F5DC',
- 'cream': 'FFFDD0',
- 'silver': 'C0C0C0',
- 'gold': 'FFD700',
- 'bronze': 'CD7F32',
- 'copper': 'B87333',
- 'navy': '000080',
- 'teal': '008080',
- 'olive': '808000',
- 'maroon': '800000',
- 'coral': 'FF7F50',
- 'salmon': 'FA8072',
- 'lime': '32CD32',
- 'mint': '98FF98',
- 'forest green': '228B22',
- 'sky blue': '87CEEB',
- 'royal blue': '4169E1',
- 'turquoise': '40E0D0',
- 'lavender': 'E6E6FA',
- 'violet': 'EE82EE',
- 'plum': 'DDA0DD',
- 'tan': 'D2B48C',
- 'chocolate': 'D2691E',
- 'charcoal': '36454F',
- 'slate': '708090',
- 'transparent': '000000', // Will need special handling
- 'natural': 'F5F5DC',
- 'wood': 'DEB887',
- };
- // Quick-select color presets (common filament colors)
- // Basic colors shown by default
- const QUICK_COLORS_BASIC = [
- { name: 'White', hex: 'FFFFFF' },
- { name: 'Black', hex: '000000' },
- { name: 'Red', hex: 'FF0000' },
- { name: 'Blue', hex: '0000FF' },
- { name: 'Green', hex: '00AA00' },
- { name: 'Yellow', hex: 'FFFF00' },
- { name: 'Orange', hex: 'FFA500' },
- { name: 'Gray', hex: '808080' },
- ];
- // Extended colors shown when expanded
- const QUICK_COLORS_EXTENDED = [
- { name: 'Cyan', hex: '00FFFF' },
- { name: 'Magenta', hex: 'FF00FF' },
- { name: 'Purple', hex: '800080' },
- { name: 'Pink', hex: 'FFC0CB' },
- { name: 'Brown', hex: '8B4513' },
- { name: 'Beige', hex: 'F5F5DC' },
- { name: 'Navy', hex: '000080' },
- { name: 'Teal', hex: '008080' },
- { name: 'Lime', hex: '32CD32' },
- { name: 'Gold', hex: 'FFD700' },
- { name: 'Silver', hex: 'C0C0C0' },
- { name: 'Maroon', hex: '800000' },
- { name: 'Olive', hex: '808000' },
- { name: 'Coral', hex: 'FF7F50' },
- { name: 'Salmon', hex: 'FA8072' },
- { name: 'Turquoise', hex: '40E0D0' },
- { name: 'Violet', hex: 'EE82EE' },
- { name: 'Indigo', hex: '4B0082' },
- { name: 'Chocolate', hex: 'D2691E' },
- { name: 'Tan', hex: 'D2B48C' },
- { name: 'Slate', hex: '708090' },
- { name: 'Charcoal', hex: '36454F' },
- { name: 'Ivory', hex: 'FFFFF0' },
- { name: 'Cream', hex: 'FFFDD0' },
- ];
- // Try to convert color name to hex
- function colorNameToHex(name: string): string | null {
- const normalized = name.toLowerCase().trim();
- return COLOR_NAME_MAP[normalized] || null;
- }
- // Extract printer model from preset name suffix "@BBL X1C 0.4 nozzle" → "X1C"
- function extractPresetModel(name: string): string | null {
- const atIdx = name.indexOf('@');
- if (atIdx < 0) return null;
- const suffix = name.slice(atIdx + 1).trim();
- const bblMatch = suffix.match(/^BBL\s+(.+?)(?:\s+[\d.]+\s*nozzle)?$/i);
- if (bblMatch) return bblMatch[1].trim();
- return null;
- }
- export function ConfigureAmsSlotModal({
- isOpen,
- onClose,
- printerId,
- slotInfo,
- nozzleDiameter = '0.4',
- printerModel,
- onSuccess,
- fullScreen,
- }: ConfigureAmsSlotModalProps) {
- const { t } = useTranslation();
- const [selectedPresetId, setSelectedPresetId] = useState<string>('');
- const [selectedKProfile, setSelectedKProfile] = useState<KProfile | null>(null);
- const [colorHex, setColorHex] = useState<string>(''); // Just the 6-char hex, no alpha
- const [colorInput, setColorInput] = useState<string>(''); // User's text input (name or hex)
- const [searchQuery, setSearchQuery] = useState('');
- const [showSuccess, setShowSuccess] = useState(false);
- const [showExtendedColors, setShowExtendedColors] = useState(false);
- const scrolledToRef = useRef<string>('');
- // Fetch cloud settings (gracefully handle 401 when logged out)
- const { data: cloudSettings, isLoading: settingsLoading, isError: cloudError } = useQuery({
- queryKey: ['cloudSettings'],
- queryFn: () => api.getCloudSettings(),
- enabled: isOpen,
- retry: false,
- });
- // Fetch local presets
- const { data: localPresets, isLoading: localLoading } = useQuery({
- queryKey: ['localPresets'],
- queryFn: () => api.getLocalPresets(),
- enabled: isOpen,
- });
- // Fetch built-in filament names (static fallback)
- const { data: builtinFilaments, isLoading: builtinLoading } = useQuery({
- queryKey: ['builtinFilaments'],
- queryFn: () => api.getBuiltinFilaments(),
- enabled: isOpen,
- staleTime: Infinity,
- });
- // Fetch K profiles
- const { data: kprofilesData, isLoading: kprofilesLoading } = useQuery({
- queryKey: ['kprofiles', printerId, nozzleDiameter],
- queryFn: () => api.getKProfiles(printerId, nozzleDiameter),
- enabled: isOpen && !!printerId,
- });
- // Fetch color catalog
- const { data: colorCatalog } = useQuery({
- queryKey: ['colorCatalog'],
- queryFn: () => api.getColorCatalog(),
- enabled: isOpen,
- staleTime: Infinity,
- });
- // Configure slot mutation
- const configureMutation = useMutation({
- mutationFn: async () => {
- if (!selectedPresetId) throw new Error('No filament preset selected');
- // Determine preset source
- const isLocal = selectedPresetId.startsWith('local_');
- const isBuiltin = selectedPresetId.startsWith('builtin_');
- const localId = isLocal ? parseInt(selectedPresetId.replace('local_', ''), 10) : null;
- const builtinFilamentId = isBuiltin ? selectedPresetId.replace('builtin_', '') : null;
- const localPreset = isLocal
- ? localPresets?.filament.find(p => p.id === localId)
- : null;
- const builtinPreset = isBuiltin
- ? builtinFilaments?.find(b => b.filament_id === builtinFilamentId)
- : null;
- // Get the selected cloud preset details (null for local/builtin presets)
- const selectedPreset = (!isLocal && !isBuiltin)
- ? cloudSettings?.filament.find(p => p.setting_id === selectedPresetId)
- : null;
- if (!isLocal && !isBuiltin && !selectedPreset) throw new Error('Selected preset not found');
- if (isLocal && !localPreset) throw new Error('Selected local preset not found');
- if (isBuiltin && !builtinPreset) throw new Error('Selected builtin preset not found');
- // Parse the preset name for filament info
- const presetName = isLocal ? localPreset!.name : isBuiltin ? builtinPreset!.name : selectedPreset!.name;
- const parsed = parsePresetName(presetName);
- // Get cali_idx from selected K profile's slot_id (-1 = use default 0.020)
- const caliIdx = selectedKProfile?.slot_id ?? -1;
- // Use custom color if set, otherwise use current slot color or default
- const color = colorHex || slotInfo.trayColor?.slice(0, 6) || 'FFFFFF';
- // Create the tray_sub_brands from preset name (without printer/nozzle suffix)
- const traySubBrands = presetName.replace(/@.+$/, '').trim();
- let trayInfoIdx: string;
- let settingId: string;
- // Parsed material from preset name — handles "Support for" patterns correctly.
- // Prefer this over stored filament_type which may have been parsed with old logic.
- const parsedMat = parsed.material.toUpperCase();
- if (isLocal) {
- // Local presets have no Bambu Cloud setting_id, but need a valid
- // tray_info_idx for the printer to recognize the filament type.
- // Map the material type to the closest generic Bambu filament ID.
- const material = (MATERIAL_TYPES.includes(parsedMat) ? parsedMat : localPreset?.filament_type || parsed.material || '').toUpperCase();
- const GENERIC_IDS: Record<string, string> = {
- 'PLA': 'GFL99', 'PLA-CF': 'GFL98', 'PLA SILK': 'GFL96', 'PLA HIGH SPEED': 'GFL95',
- 'PETG': 'GFG99', 'PETG HF': 'GFG96', 'PETG-CF': 'GFG98', 'PCTG': 'GFG97',
- 'ABS': 'GFB99', 'ASA': 'GFB98',
- 'PC': 'GFC99',
- 'PA': 'GFN99', 'PA-CF': 'GFN98', 'NYLON': 'GFN99',
- 'TPU': 'GFU99',
- 'PVA': 'GFS99', 'HIPS': 'GFS98',
- 'PE': 'GFP99', 'PP': 'GFP97',
- };
- // Try exact match first, then base material (strip suffixes like "-CF", "+", " HF")
- trayInfoIdx = GENERIC_IDS[material]
- || GENERIC_IDS[material.replace(/[-\s]?CF$/, '')]
- || GENERIC_IDS[material.replace(/\+$/, '')]
- || GENERIC_IDS[material.split(/[-\s]/)[0]]
- || '';
- settingId = '';
- } else if (isBuiltin) {
- // Built-in presets use the filament_id directly as tray_info_idx
- trayInfoIdx = builtinFilamentId!;
- settingId = '';
- } else {
- trayInfoIdx = convertToTrayInfoIdx(selectedPresetId);
- settingId = selectedPresetId;
- // User cloud presets may carry a distinct filament_id in the cloud detail
- // (e.g. "P285e239"); prefer it when present. Never fall back to base_id —
- // that collapses custom presets to the inherited generic's filament_id and
- // makes the slicer resolve the slot to "Generic …" instead (#1053).
- if (!selectedPresetId.startsWith('GFS')) {
- try {
- const detail = await api.getCloudSettingDetail(selectedPresetId);
- if (detail.filament_id) {
- trayInfoIdx = detail.filament_id;
- }
- } catch (e) {
- console.warn('Failed to fetch preset detail for filament_id:', e);
- }
- }
- }
- // Default temp range — use local preset core fields if available
- let tempMin = isLocal && localPreset?.nozzle_temp_min ? localPreset.nozzle_temp_min : 190;
- let tempMax = isLocal && localPreset?.nozzle_temp_max ? localPreset.nozzle_temp_max : 230;
- if (!isLocal || isBuiltin || (!localPreset?.nozzle_temp_min && !localPreset?.nozzle_temp_max)) {
- // Fall back to material-based defaults (prefer parsed material for "Support for" handling)
- const material = (isLocal
- ? (MATERIAL_TYPES.includes(parsedMat) ? parsedMat : localPreset?.filament_type || parsed.material || '')
- : parsed.material).toUpperCase();
- if (material.includes('PLA')) {
- tempMin = 190;
- tempMax = 230;
- } else if (material.includes('PETG')) {
- tempMin = 220;
- tempMax = 260;
- } else if (material.includes('ABS')) {
- tempMin = 240;
- tempMax = 280;
- } else if (material.includes('ASA')) {
- tempMin = 240;
- tempMax = 280;
- } else if (material.includes('TPU')) {
- tempMin = 200;
- tempMax = 240;
- } else if (material === 'PCTG') {
- tempMin = 220;
- tempMax = 260;
- } else if (material.includes('PC')) {
- tempMin = 260;
- tempMax = 300;
- } else if (material.includes('PA') || material.includes('NYLON')) {
- tempMin = 250;
- tempMax = 290;
- }
- }
- // Parse K value from selected profile
- const kValue = selectedKProfile?.k_value ? parseFloat(selectedKProfile.k_value) : 0;
- // Determine tray_type: prefer parsed material from preset name (handles "Support for"
- // patterns correctly) over stored filament_type which may have been parsed with old logic.
- const trayType = isLocal
- ? (MATERIAL_TYPES.includes(parsedMat) ? parsedMat : localPreset?.filament_type || parsed.material || 'PLA')
- : (parsed.material || 'PLA');
- // Configure the slot via MQTT
- const result = await api.configureAmsSlot(printerId, slotInfo.amsId, slotInfo.trayId, {
- tray_info_idx: trayInfoIdx,
- tray_type: trayType,
- tray_sub_brands: traySubBrands,
- tray_color: color + 'FF', // Add alpha
- nozzle_temp_min: tempMin,
- nozzle_temp_max: tempMax,
- cali_idx: caliIdx,
- nozzle_diameter: nozzleDiameter,
- setting_id: settingId, // Full setting ID for slicer compatibility (empty for local)
- // Pass K profile's filament_id and setting_id for proper linking
- kprofile_filament_id: selectedKProfile?.filament_id,
- kprofile_setting_id: selectedKProfile?.setting_id || undefined,
- // Also pass the K value directly for extrusion_cali_set command
- k_value: kValue,
- });
- // Save the preset mapping so we can display the correct name in the UI
- // This is needed because user presets use filament_id (e.g., P285e239) as tray_info_idx,
- // which can't be resolved to a name via the filamentInfo API
- const mappingPresetId = isLocal ? `local_${localId}` : isBuiltin ? `builtin_${builtinFilamentId}` : selectedPresetId;
- const mappingSource = isLocal ? 'local' : isBuiltin ? 'builtin' : 'cloud';
- try {
- await api.saveSlotPreset(printerId, slotInfo.amsId, slotInfo.trayId, mappingPresetId, traySubBrands, mappingSource);
- } catch (e) {
- console.warn('Failed to save slot preset mapping:', e);
- // Don't fail the whole operation - slot was configured successfully
- }
- return result;
- },
- onSuccess: () => {
- setShowSuccess(true);
- onSuccess?.();
- // Close after showing success briefly
- setTimeout(() => {
- setShowSuccess(false);
- onClose();
- }, 1500);
- },
- });
- // Reset slot mutation
- const resetMutation = useMutation({
- mutationFn: async () => {
- return api.resetAmsSlot(printerId, slotInfo.amsId, slotInfo.trayId);
- },
- onSuccess: () => {
- setShowSuccess(true);
- onSuccess?.();
- setTimeout(() => {
- setShowSuccess(false);
- onClose();
- }, 1500);
- },
- });
- // Unified preset item for the list (cloud + local + builtin fallback)
- type PresetItem = { id: string; name: string; source: 'cloud' | 'local' | 'builtin'; isUser: boolean };
- // Filter filament presets based on search (merged cloud + local + builtin)
- const filteredPresets = useMemo(() => {
- const query = searchQuery.toLowerCase();
- const items: PresetItem[] = [];
- // Collect IDs already covered by cloud and local to avoid duplicates in fallback
- const coveredIds = new Set<string>();
- // Currently-configured preset should always be shown (bypass model filter)
- const savedId = slotInfo.savedPresetId;
- const trayIdx = slotInfo.trayInfoIdx;
- // 1. Cloud presets
- if (cloudSettings?.filament) {
- for (const cp of cloudSettings.filament) {
- coveredIds.add(cp.setting_id);
- // Keep preset if it matches the slot's saved mapping or current tray_info_idx
- const isSavedPreset = savedId === cp.setting_id;
- const isCurrentPreset = isSavedPreset
- || (trayIdx && (cp.setting_id === trayIdx || convertToTrayInfoIdx(cp.setting_id) === trayIdx));
- // Search filter applies to ALL presets (including saved) — no bypass
- if (query && !cp.name.toLowerCase().includes(query)) continue;
- // Filter by printer model if set (skip for current preset)
- if (!isCurrentPreset && printerModel) {
- const presetModel = extractPresetModel(cp.name);
- if (presetModel && presetModel.toUpperCase() !== printerModel.toUpperCase()) continue;
- }
- items.push({ id: cp.setting_id, name: cp.name, source: 'cloud', isUser: isUserPreset(cp.setting_id) });
- }
- }
- // 2. Local presets (always shown — user-imported profiles work on any printer)
- if (localPresets?.filament) {
- for (const lp of localPresets.filament) {
- const localId = `local_${lp.id}`;
- if (query && !lp.name.toLowerCase().includes(query)) continue;
- items.push({ id: localId, name: lp.name, source: 'local', isUser: false });
- }
- }
- // 3. Built-in filament names (fallback — only add entries not already covered)
- if (builtinFilaments) {
- for (const bf of builtinFilaments) {
- if (coveredIds.has(bf.filament_id)) continue;
- // Convert filament_id to setting_id format for cloud compatibility
- // e.g. "GFA00" → cloud setting_id would be "GFSA00" (insert S after GF)
- const settingId = bf.filament_id.startsWith('GF')
- ? 'GFS' + bf.filament_id.slice(2)
- : bf.filament_id;
- if (coveredIds.has(settingId)) continue;
- if (!query || bf.name.toLowerCase().includes(query)) {
- items.push({ id: `builtin_${bf.filament_id}`, name: bf.name, source: 'builtin', isUser: false });
- }
- }
- }
- // Sort: cloud user presets first, then cloud built-in, then local, then builtin fallback
- return items.sort((a, b) => {
- const sourceOrder = { cloud: 0, local: 1, builtin: 2 };
- if (a.source !== b.source) return sourceOrder[a.source] - sourceOrder[b.source];
- if (a.isUser && !b.isUser) return -1;
- if (!a.isUser && b.isUser) return 1;
- return a.name.localeCompare(b.name);
- });
- }, [cloudSettings?.filament, localPresets?.filament, builtinFilaments, searchQuery, printerModel, slotInfo.savedPresetId, slotInfo.trayInfoIdx]);
- // Get full preset name for K profile filtering (brand + material, without printer suffix)
- const selectedPresetInfo = useMemo(() => {
- if (!selectedPresetId) return null;
- // Resolve the name from cloud, local, or builtin presets
- let presetName: string | null = null;
- if (selectedPresetId.startsWith('local_')) {
- const localId = parseInt(selectedPresetId.replace('local_', ''), 10);
- const lp = localPresets?.filament.find(p => p.id === localId);
- presetName = lp?.name || null;
- } else if (selectedPresetId.startsWith('builtin_')) {
- const filamentId = selectedPresetId.replace('builtin_', '');
- const bf = builtinFilaments?.find(b => b.filament_id === filamentId);
- presetName = bf?.name || null;
- } else if (cloudSettings?.filament) {
- const cp = cloudSettings.filament.find(p => p.setting_id === selectedPresetId);
- presetName = cp?.name || null;
- } else {
- // No cloud settings available
- }
- if (!presetName) {
- return null;
- }
- // Remove printer/nozzle suffix (e.g., "@BBL X1C" or "@0.4 nozzle")
- let nameWithoutSuffix = presetName.replace(/@.+$/, '').trim();
- // Strip leading "# " from custom preset names (user convention)
- if (nameWithoutSuffix.startsWith('# ')) {
- nameWithoutSuffix = nameWithoutSuffix.slice(2).trim();
- }
- const parsed = parsePresetName(nameWithoutSuffix);
- return {
- fullName: nameWithoutSuffix,
- material: parsed.material,
- brand: parsed.brand,
- };
- }, [selectedPresetId, cloudSettings?.filament, localPresets?.filament, builtinFilaments]);
- // For backwards compatibility with the label
- const selectedMaterial = selectedPresetInfo?.fullName || '';
- // Filter color catalog entries matching the selected preset's brand + material
- const catalogColors = useMemo(() => {
- if (!colorCatalog || !selectedPresetInfo) return [];
- const { fullName, brand } = selectedPresetInfo;
- // Try to find colors matching the full preset name (e.g., "PLA Metal")
- // The catalog uses the variant as part of the material field (e.g., material="PLA Metal")
- // Extract the full material+variant from the preset name
- const materialVariant = fullName.replace(/^(Bambu\s*(Lab)?|eSUN|Polymaker|Overture|Sunlu|Hatchbox)\s*/i, '').trim();
- return colorCatalog.filter(entry => {
- const entryMaterial = (entry.material || '').toUpperCase();
- const entryManufacturer = entry.manufacturer.toUpperCase();
- // Match material: try full material+variant first, then just material type
- const materialMatch = entryMaterial === materialVariant.toUpperCase()
- || entryMaterial.includes(materialVariant.toUpperCase())
- || materialVariant.toUpperCase().includes(entryMaterial);
- if (!materialMatch) return false;
- // If brand is present, also match manufacturer
- if (brand) {
- const upperBrand = brand.toUpperCase();
- // Fuzzy match: "Bambu" matches "Bambu Lab", etc.
- if (!entryManufacturer.includes(upperBrand) && !upperBrand.includes(entryManufacturer)) {
- return false;
- }
- }
- return true;
- });
- }, [colorCatalog, selectedPresetInfo]);
- const matchingKProfiles = useMemo(() => {
- if (!kprofilesData?.profiles || !selectedPresetInfo) return [];
- const { fullName, material, brand } = selectedPresetInfo;
- const upperFullName = fullName.toUpperCase();
- const upperMaterial = material.toUpperCase();
- const upperBrand = brand.toUpperCase();
- // Material must be at least 2 chars to avoid false positives
- if (!upperMaterial || upperMaterial.length < 2) return [];
- // Filter profiles - require brand match if brand is present in selected preset
- const filtered = kprofilesData.profiles.filter(p => {
- const profileName = p.name.toUpperCase();
- // If the selected preset has a brand (e.g., "Azurefilm PLA Wood"),
- // only show profiles that match the brand
- if (upperBrand) {
- // Must contain the brand name
- if (!profileName.includes(upperBrand)) {
- return false;
- }
- // And must contain the material type
- if (!profileName.includes(upperMaterial)) {
- return false;
- }
- return true;
- }
- // No brand in selected preset - match on full name or material
- // Priority 1: Exact match with full name
- if (profileName.includes(upperFullName)) {
- return true;
- }
- // Priority 2: Material type match (only when no brand specified)
- if (profileName.includes(upperMaterial)) {
- return true;
- }
- // Check for common material aliases
- const aliases: Record<string, string[]> = {
- 'NYLON': ['PA', 'PA-CF', 'PA6'],
- 'PA': ['NYLON'],
- };
- const materialAliases = aliases[upperMaterial] || [];
- for (const alias of materialAliases) {
- if (profileName.includes(alias)) {
- return true;
- }
- }
- return false;
- });
- // Deduplicate profiles with same name and k_value (multi-nozzle printers have duplicates)
- // Prefer the profile matching the slot's extruder (e.g. ext-R uses extruder 0, ext-L uses extruder 1)
- const seen = new Map<string, KProfile>();
- for (const profile of filtered) {
- const key = `${profile.name}|${profile.k_value}`;
- const existing = seen.get(key);
- if (!existing) {
- seen.set(key, profile);
- } else if (slotInfo.extruderId !== undefined && profile.extruder_id === slotInfo.extruderId && existing.extruder_id !== slotInfo.extruderId) {
- // Replace with profile matching slot's extruder
- seen.set(key, profile);
- }
- }
- return Array.from(seen.values());
- }, [kprofilesData?.profiles, selectedPresetInfo, slotInfo.extruderId]);
- // Pre-select current profile when modal opens, reset when closes
- useEffect(() => {
- if (isOpen) {
- // Pre-populate from saved preset mapping (most reliable)
- if (slotInfo.savedPresetId) {
- setSelectedPresetId(slotInfo.savedPresetId);
- } else if (slotInfo.trayInfoIdx && cloudSettings?.filament) {
- // Fallback: try to match by tray_info_idx in cloud presets
- // First try exact match on setting_id
- let currentPreset = cloudSettings.filament.find(
- p => p.setting_id === slotInfo.trayInfoIdx
- );
- // Then try matching by converting setting_id → filament_id format
- if (!currentPreset) {
- currentPreset = cloudSettings.filament.find(
- p => convertToTrayInfoIdx(p.setting_id) === slotInfo.trayInfoIdx
- );
- }
- if (currentPreset) {
- setSelectedPresetId(currentPreset.setting_id);
- }
- } else if (slotInfo.trayInfoIdx && builtinFilaments?.length) {
- // Last resort: match trayInfoIdx against builtin presets
- const trayIdx = slotInfo.trayInfoIdx;
- const match = builtinFilaments.find(bf => bf.filament_id === trayIdx);
- if (match) {
- setSelectedPresetId(`builtin_${match.filament_id}`);
- }
- }
- // Pre-populate color from current slot (black is valid — empty slots don't pass trayColor)
- if (slotInfo.trayColor) {
- const hex = slotInfo.trayColor.slice(0, 6);
- if (hex) {
- setColorHex(hex);
- }
- }
- } else {
- // Reset when modal closes
- setSelectedPresetId('');
- setSelectedKProfile(null);
- setColorHex('');
- setColorInput('');
- setSearchQuery('');
- setShowSuccess(false);
- scrolledToRef.current = '';
- }
- }, [isOpen, slotInfo.savedPresetId, slotInfo.trayInfoIdx, slotInfo.trayColor, cloudSettings?.filament, builtinFilaments]);
- // Auto-select best matching K profile when preset changes
- useEffect(() => {
- if (matchingKProfiles.length > 0) {
- // Prefer the currently-active K-profile (by cali_idx) if available
- if (slotInfo.caliIdx != null && slotInfo.caliIdx > 0) {
- const active = matchingKProfiles.find(p => p.slot_id === slotInfo.caliIdx);
- if (active) {
- setSelectedKProfile(active);
- return;
- }
- }
- // Fallback: first matching profile
- setSelectedKProfile(matchingKProfiles[0]);
- } else {
- setSelectedKProfile(null);
- }
- }, [selectedPresetId, matchingKProfiles, slotInfo.caliIdx]);
- // Escape key handler
- const handleKeyDown = useCallback((e: KeyboardEvent) => {
- if (e.key === 'Escape') {
- onClose();
- }
- }, [onClose]);
- useEffect(() => {
- if (isOpen) {
- document.addEventListener('keydown', handleKeyDown);
- return () => document.removeEventListener('keydown', handleKeyDown);
- }
- }, [isOpen, handleKeyDown]);
- const isLoading = (settingsLoading && !cloudError) || localLoading || builtinLoading || kprofilesLoading;
- // Scroll selected preset into view when data finishes loading or the selection changes.
- // Uses a ref guard so scrollIntoView only fires once per selection, preventing the
- // infinite scroll loop that occurred on Windows with inline callback refs.
- useEffect(() => {
- if (!isLoading && selectedPresetId && selectedPresetId !== scrolledToRef.current) {
- const raf = requestAnimationFrame(() => {
- const modal = document.querySelector('[class*="fixed inset-0 z-50"]');
- const el = modal?.querySelector(`[data-preset-id="${CSS.escape(selectedPresetId)}"]`);
- if (el) {
- scrolledToRef.current = selectedPresetId;
- el.scrollIntoView({ block: 'nearest' });
- }
- });
- return () => cancelAnimationFrame(raf);
- }
- }, [selectedPresetId, isLoading]);
- if (!isOpen) return null;
- const canSave = selectedPresetId && !configureMutation.isPending;
- // Get display color (custom or slot default)
- const displayColor = colorHex || slotInfo.trayColor?.slice(0, 6) || 'FFFFFF';
- return (
- <div className={`fixed inset-0 z-50 flex ${fullScreen ? '' : 'items-center justify-center'}`}>
- {/* Backdrop */}
- {!fullScreen && (
- <div
- className="absolute inset-0 bg-black/60 backdrop-blur-sm"
- onClick={onClose}
- />
- )}
- {/* Modal */}
- <div className={fullScreen
- ? 'relative w-full h-full bg-bambu-dark-secondary flex flex-col'
- : 'relative w-full max-w-lg mx-4 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-xl shadow-2xl'
- }>
- {/* Header */}
- <div className="flex items-center justify-between p-4 border-b border-bambu-dark-tertiary shrink-0">
- <div className="flex items-center gap-2">
- <Settings2 className="w-5 h-5 text-bambu-blue" />
- <h2 className="text-lg font-semibold text-white">{t('configureAmsSlot.title')}</h2>
- {/* Inline slot info in fullScreen mode */}
- {fullScreen && (
- <div className="flex items-center gap-2 ml-4 text-sm text-bambu-gray">
- <span className="text-white/30">|</span>
- {slotInfo.trayColor && (
- <span
- className="w-4 h-4 rounded-full border border-black/20"
- style={{ backgroundColor: `#${slotInfo.trayColor.slice(0, 6)}` }}
- />
- )}
- <span className="text-white/70">
- {t('configureAmsSlot.slotLabel', { ams: getAmsLabel(slotInfo.amsId, slotInfo.trayCount), slot: slotInfo.trayId + 1 })}
- </span>
- {slotInfo.traySubBrands && (
- <span>({slotInfo.traySubBrands})</span>
- )}
- </div>
- )}
- </div>
- <button
- onClick={onClose}
- className="p-1 text-bambu-gray hover:text-white rounded transition-colors"
- >
- <X className="w-5 h-5" />
- </button>
- </div>
- {/* Content */}
- <div className={`p-4 overflow-y-auto ${fullScreen ? 'flex-1 min-h-0' : 'space-y-4 max-h-[60vh]'}`}>
- {/* Success overlay */}
- {showSuccess && (
- <div className="absolute inset-0 bg-bambu-dark-secondary/95 z-10 flex items-center justify-center rounded-xl">
- <div className="text-center space-y-3">
- <CheckCircle2 className="w-16 h-16 text-bambu-green mx-auto" />
- <p className="text-lg font-semibold text-white">{t('configureAmsSlot.slotConfigured')}</p>
- <p className="text-sm text-bambu-gray">{t('configureAmsSlot.settingsSentToPrinter')}</p>
- </div>
- </div>
- )}
- {/* Slot info */}
- {!fullScreen && (
- <div className="p-3 bg-bambu-dark rounded-lg border border-bambu-dark-tertiary">
- <p className="text-xs text-bambu-gray mb-1">{t('configureAmsSlot.configuringSlot')}</p>
- <div className="flex items-center gap-2">
- {slotInfo.trayColor && (
- <span
- className="w-4 h-4 rounded-full border border-black/20"
- style={{ backgroundColor: `#${slotInfo.trayColor.slice(0, 6)}` }}
- />
- )}
- <span className="text-white font-medium">
- {t('configureAmsSlot.slotLabel', { ams: getAmsLabel(slotInfo.amsId, slotInfo.trayCount), slot: slotInfo.trayId + 1 })}
- </span>
- {slotInfo.traySubBrands && (
- <span className="text-bambu-gray">({slotInfo.traySubBrands})</span>
- )}
- </div>
- </div>
- )}
- {isLoading ? (
- <div className="flex justify-center py-8">
- <Loader2 className="w-6 h-6 text-bambu-green animate-spin" />
- </div>
- ) : fullScreen ? (
- /* Two-column layout for kiosk display */
- <div className="flex gap-4 h-full">
- {/* Left column: Filament preset list (takes full height) */}
- <div className="w-1/2 flex flex-col min-h-0">
- <label className="block text-sm text-bambu-gray mb-2">
- {t('configureAmsSlot.filamentProfile')} <span className="text-red-400">*</span>
- </label>
- <input
- type="text"
- placeholder={t('configureAmsSlot.searchPresets')}
- value={searchQuery}
- onChange={(e) => setSearchQuery(e.target.value)}
- className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white placeholder:text-bambu-gray focus:border-bambu-green focus:outline-none mb-2 shrink-0"
- />
- <div className="flex-1 min-h-0 overflow-y-auto space-y-1">
- {filteredPresets.length === 0 ? (
- <p className="text-center py-4 text-bambu-gray">
- {(cloudSettings?.filament?.length === 0 && !localPresets?.filament?.length)
- ? t('configureAmsSlot.noPresetsAvailable')
- : t('configureAmsSlot.noMatchingPresets')}
- </p>
- ) : (
- filteredPresets.map((preset) => (
- <button
- key={preset.id}
- data-preset-id={preset.id}
- onClick={() => setSelectedPresetId(preset.id)}
- className={`w-full p-2 rounded-lg border text-left transition-colors ${
- selectedPresetId === preset.id
- ? 'bg-bambu-green/20 border-bambu-green'
- : 'bg-bambu-dark border-bambu-dark-tertiary hover:border-bambu-gray'
- }`}
- >
- <div className="flex items-center justify-between">
- <span className="text-white text-sm truncate">{preset.name}</span>
- <div className="flex items-center gap-1 flex-shrink-0">
- {preset.source === 'local' && (
- <span className="text-xs px-1.5 py-0.5 rounded bg-green-500/20 text-green-400">
- {t('profiles.localProfiles.badge')}
- </span>
- )}
- {preset.source === 'builtin' && (
- <span className="text-xs px-1.5 py-0.5 rounded bg-amber-500/20 text-amber-400">
- {t('configureAmsSlot.builtin')}
- </span>
- )}
- {preset.isUser && (
- <span className="text-xs px-1.5 py-0.5 rounded bg-bambu-blue/20 text-bambu-blue">
- {t('configureAmsSlot.custom')}
- </span>
- )}
- </div>
- </div>
- </button>
- ))
- )}
- </div>
- </div>
- {/* Right column: K Profile + Color */}
- <div className="w-1/2 flex flex-col gap-4 min-h-0 overflow-y-auto">
- {/* K Profile Select */}
- <div>
- <label className="block text-sm text-bambu-gray mb-2">
- {t('configureAmsSlot.kProfileLabel')}
- {selectedMaterial && (
- <span className="ml-2 text-xs text-bambu-blue">
- {t('configureAmsSlot.filteringFor', { material: selectedMaterial })}
- </span>
- )}
- </label>
- {matchingKProfiles.length > 0 ? (
- <div className="relative">
- <select
- value={selectedKProfile?.name || ''}
- onChange={(e) => {
- const profile = matchingKProfiles.find(p => p.name === e.target.value);
- setSelectedKProfile(profile || null);
- }}
- 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 appearance-none pr-10"
- >
- <option value="">{t('configureAmsSlot.noKProfile')}</option>
- {matchingKProfiles.map((profile) => (
- <option key={`${profile.name}-${profile.extruder_id}`} value={profile.name}>
- {profile.name} (K={profile.k_value})
- </option>
- ))}
- </select>
- <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
- </div>
- ) : selectedPresetId ? (
- <p className="text-sm text-bambu-gray italic py-2">
- {t('configureAmsSlot.noMatchingKProfiles')}
- </p>
- ) : (
- <span className="inline-block text-xs px-2 py-1 rounded bg-amber-500/20 text-amber-400 border border-amber-500/30">
- {t('configureAmsSlot.selectFilamentFirst')}
- </span>
- )}
- {selectedKProfile && (
- <p className="text-xs text-bambu-green mt-1">
- {t('configureAmsSlot.kFromCalibration', { value: selectedKProfile.k_value })}
- </p>
- )}
- </div>
- {/* Custom color */}
- <div>
- <label className="block text-sm text-bambu-gray mb-2">
- {t('configureAmsSlot.customColorLabel')}
- </label>
- {catalogColors.length > 0 && (
- <div className="mb-3">
- <p className="text-xs text-bambu-gray mb-1.5">
- {t('configureAmsSlot.presetColors', { name: selectedPresetInfo?.fullName })}
- </p>
- <div className="flex flex-wrap gap-1.5">
- {catalogColors.map((entry) => (
- <button
- key={entry.id}
- onClick={() => {
- const hex = entry.hex_color.replace('#', '').toUpperCase();
- setColorHex(hex);
- setColorInput(entry.color_name);
- }}
- className={`h-7 px-2 rounded-md border-2 transition-all flex items-center gap-1.5 ${
- colorHex === entry.hex_color.replace('#', '').toUpperCase()
- ? 'border-bambu-green scale-105'
- : 'border-white/20 hover:border-white/40'
- }`}
- title={entry.color_name}
- >
- <span
- className="w-4 h-4 rounded-full border border-black/20 flex-shrink-0"
- style={{ backgroundColor: entry.hex_color }}
- />
- <span className="text-xs text-white/80 whitespace-nowrap">{entry.color_name}</span>
- </button>
- ))}
- </div>
- </div>
- )}
- <div className="flex flex-wrap gap-1.5 mb-2">
- {QUICK_COLORS_BASIC.map((color) => (
- <button
- key={color.hex}
- onClick={() => {
- setColorHex(color.hex);
- setColorInput(color.name);
- }}
- className={`w-7 h-7 rounded-md border-2 transition-all ${
- colorHex === color.hex
- ? 'border-bambu-green scale-110'
- : 'border-white/20 hover:border-white/40'
- }`}
- style={{ backgroundColor: `#${color.hex}` }}
- title={color.name}
- />
- ))}
- <button
- onClick={() => setShowExtendedColors(!showExtendedColors)}
- className="w-7 h-7 rounded-md border-2 border-white/20 hover:border-white/40 flex items-center justify-center text-white/60 hover:text-white/80 transition-all text-xs"
- title={showExtendedColors ? t('configureAmsSlot.showLessColors') : t('configureAmsSlot.showMoreColors')}
- >
- {showExtendedColors ? '−' : '+'}
- </button>
- </div>
- {showExtendedColors && (
- <div className="flex flex-wrap gap-1.5 mb-2">
- {QUICK_COLORS_EXTENDED.map((color) => (
- <button
- key={color.hex}
- onClick={() => {
- setColorHex(color.hex);
- setColorInput(color.name);
- }}
- className={`w-7 h-7 rounded-md border-2 transition-all ${
- colorHex === color.hex
- ? 'border-bambu-green scale-110'
- : 'border-white/20 hover:border-white/40'
- }`}
- style={{ backgroundColor: `#${color.hex}` }}
- title={color.name}
- />
- ))}
- </div>
- )}
- <div className="flex gap-2 items-center">
- <div
- className="w-10 h-10 rounded-lg border-2 border-white/20 flex-shrink-0"
- style={{ backgroundColor: `#${displayColor}` }}
- />
- <input
- type="text"
- placeholder={t('configureAmsSlot.colorPlaceholder')}
- value={colorInput}
- onChange={(e) => {
- const input = e.target.value;
- setColorInput(input);
- const nameHex = colorNameToHex(input);
- if (nameHex) {
- setColorHex(nameHex);
- } else {
- const cleaned = input.replace(/[^0-9A-Fa-f]/g, '').toUpperCase();
- if (cleaned.length === 6) {
- setColorHex(cleaned);
- } else if (cleaned.length === 3) {
- setColorHex(cleaned.split('').map(c => c + c).join(''));
- }
- }
- }}
- className="flex-1 px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white placeholder:text-bambu-gray focus:border-bambu-green focus:outline-none text-sm"
- />
- {colorHex && (
- <button
- onClick={() => {
- setColorHex('');
- setColorInput('');
- }}
- className="px-2 py-1 text-xs text-bambu-gray hover:text-white bg-bambu-dark-tertiary rounded"
- title={t('configureAmsSlot.clearCustomColor')}
- >
- {t('configureAmsSlot.clear')}
- </button>
- )}
- </div>
- {colorHex && (
- <p className="text-xs text-bambu-gray mt-1.5">
- {t('configureAmsSlot.hexLabel', { hex: colorHex })}
- </p>
- )}
- </div>
- </div>
- </div>
- ) : (
- <>
- {/* Filament Profile Select */}
- <div>
- <label className="block text-sm text-bambu-gray mb-2">
- {t('configureAmsSlot.filamentProfile')} <span className="text-red-400">*</span>
- </label>
- <div className="relative">
- <input
- type="text"
- placeholder={t('configureAmsSlot.searchPresets')}
- value={searchQuery}
- onChange={(e) => setSearchQuery(e.target.value)}
- className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white placeholder:text-bambu-gray focus:border-bambu-green focus:outline-none mb-2"
- />
- <div className="max-h-48 overflow-y-auto space-y-1">
- {filteredPresets.length === 0 ? (
- <p className="text-center py-4 text-bambu-gray">
- {(cloudSettings?.filament?.length === 0 && !localPresets?.filament?.length)
- ? t('configureAmsSlot.noPresetsAvailable')
- : t('configureAmsSlot.noMatchingPresets')}
- </p>
- ) : (
- filteredPresets.map((preset) => (
- <button
- key={preset.id}
- data-preset-id={preset.id}
- onClick={() => setSelectedPresetId(preset.id)}
- className={`w-full p-2 rounded-lg border text-left transition-colors ${
- selectedPresetId === preset.id
- ? 'bg-bambu-green/20 border-bambu-green'
- : 'bg-bambu-dark border-bambu-dark-tertiary hover:border-bambu-gray'
- }`}
- >
- <div className="flex items-center justify-between">
- <span className="text-white text-sm truncate">{preset.name}</span>
- <div className="flex items-center gap-1 flex-shrink-0">
- {preset.source === 'local' && (
- <span className="text-xs px-1.5 py-0.5 rounded bg-green-500/20 text-green-400">
- {t('profiles.localProfiles.badge')}
- </span>
- )}
- {preset.source === 'builtin' && (
- <span className="text-xs px-1.5 py-0.5 rounded bg-amber-500/20 text-amber-400">
- {t('configureAmsSlot.builtin')}
- </span>
- )}
- {preset.isUser && (
- <span className="text-xs px-1.5 py-0.5 rounded bg-bambu-blue/20 text-bambu-blue">
- {t('configureAmsSlot.custom')}
- </span>
- )}
- </div>
- </div>
- </button>
- ))
- )}
- </div>
- </div>
- </div>
- {/* K Profile Select */}
- <div>
- <label className="block text-sm text-bambu-gray mb-2">
- {t('configureAmsSlot.kProfileLabel')}
- {selectedMaterial && (
- <span className="ml-2 text-xs text-bambu-blue">
- {t('configureAmsSlot.filteringFor', { material: selectedMaterial })}
- </span>
- )}
- </label>
- {matchingKProfiles.length > 0 ? (
- <div className="relative">
- <select
- value={selectedKProfile?.name || ''}
- onChange={(e) => {
- const profile = matchingKProfiles.find(p => p.name === e.target.value);
- setSelectedKProfile(profile || null);
- }}
- 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 appearance-none pr-10"
- >
- <option value="">{t('configureAmsSlot.noKProfile')}</option>
- {matchingKProfiles.map((profile) => (
- <option key={`${profile.name}-${profile.extruder_id}`} value={profile.name}>
- {profile.name} (K={profile.k_value})
- </option>
- ))}
- </select>
- <ChevronDown className="absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
- </div>
- ) : selectedPresetId ? (
- <p className="text-sm text-bambu-gray italic py-2">
- {t('configureAmsSlot.noMatchingKProfiles')}
- </p>
- ) : (
- <span className="inline-block text-xs px-2 py-1 rounded bg-amber-500/20 text-amber-400 border border-amber-500/30">
- {t('configureAmsSlot.selectFilamentFirst')}
- </span>
- )}
- {selectedKProfile && (
- <p className="text-xs text-bambu-green mt-1">
- {t('configureAmsSlot.kFromCalibration', { value: selectedKProfile.k_value })}
- </p>
- )}
- </div>
- {/* Optional: Custom color */}
- <div>
- <label className="block text-sm text-bambu-gray mb-2">
- {t('configureAmsSlot.customColorLabel')}
- </label>
- {/* Catalog colors matching selected preset */}
- {catalogColors.length > 0 && (
- <div className="mb-3">
- <p className="text-xs text-bambu-gray mb-1.5">
- {t('configureAmsSlot.presetColors', { name: selectedPresetInfo?.fullName })}
- </p>
- <div className="flex flex-wrap gap-1.5">
- {catalogColors.map((entry) => (
- <button
- key={entry.id}
- onClick={() => {
- const hex = entry.hex_color.replace('#', '').toUpperCase();
- setColorHex(hex);
- setColorInput(entry.color_name);
- }}
- className={`h-7 px-2 rounded-md border-2 transition-all flex items-center gap-1.5 ${
- colorHex === entry.hex_color.replace('#', '').toUpperCase()
- ? 'border-bambu-green scale-105'
- : 'border-white/20 hover:border-white/40'
- }`}
- title={entry.color_name}
- >
- <span
- className="w-4 h-4 rounded-full border border-black/20 flex-shrink-0"
- style={{ backgroundColor: entry.hex_color }}
- />
- <span className="text-xs text-white/80 whitespace-nowrap">{entry.color_name}</span>
- </button>
- ))}
- </div>
- </div>
- )}
- {/* Quick color buttons */}
- <div className="flex flex-wrap gap-1.5 mb-2">
- {QUICK_COLORS_BASIC.map((color) => (
- <button
- key={color.hex}
- onClick={() => {
- setColorHex(color.hex);
- setColorInput(color.name);
- }}
- className={`w-7 h-7 rounded-md border-2 transition-all ${
- colorHex === color.hex
- ? 'border-bambu-green scale-110'
- : 'border-white/20 hover:border-white/40'
- }`}
- style={{ backgroundColor: `#${color.hex}` }}
- title={color.name}
- />
- ))}
- <button
- onClick={() => setShowExtendedColors(!showExtendedColors)}
- className="w-7 h-7 rounded-md border-2 border-white/20 hover:border-white/40 flex items-center justify-center text-white/60 hover:text-white/80 transition-all text-xs"
- title={showExtendedColors ? t('configureAmsSlot.showLessColors') : t('configureAmsSlot.showMoreColors')}
- >
- {showExtendedColors ? '−' : '+'}
- </button>
- </div>
- {/* Extended colors (collapsible) */}
- {showExtendedColors && (
- <div className="flex flex-wrap gap-1.5 mb-2">
- {QUICK_COLORS_EXTENDED.map((color) => (
- <button
- key={color.hex}
- onClick={() => {
- setColorHex(color.hex);
- setColorInput(color.name);
- }}
- className={`w-7 h-7 rounded-md border-2 transition-all ${
- colorHex === color.hex
- ? 'border-bambu-green scale-110'
- : 'border-white/20 hover:border-white/40'
- }`}
- style={{ backgroundColor: `#${color.hex}` }}
- title={color.name}
- />
- ))}
- </div>
- )}
- {/* Color input: name or hex */}
- <div className="flex gap-2 items-center">
- <div
- className="w-10 h-10 rounded-lg border-2 border-white/20 flex-shrink-0"
- style={{ backgroundColor: `#${displayColor}` }}
- />
- <input
- type="text"
- placeholder={t('configureAmsSlot.colorPlaceholder')}
- value={colorInput}
- onChange={(e) => {
- const input = e.target.value;
- setColorInput(input);
- // Try to parse as color name first
- const nameHex = colorNameToHex(input);
- if (nameHex) {
- setColorHex(nameHex);
- } else {
- // Try to parse as hex code
- const cleaned = input.replace(/[^0-9A-Fa-f]/g, '').toUpperCase();
- if (cleaned.length === 6) {
- setColorHex(cleaned);
- } else if (cleaned.length === 3) {
- // Expand shorthand hex (e.g., F00 -> FF0000)
- setColorHex(cleaned.split('').map(c => c + c).join(''));
- }
- }
- }}
- className="flex-1 px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white placeholder:text-bambu-gray focus:border-bambu-green focus:outline-none text-sm"
- />
- {colorHex && (
- <button
- onClick={() => {
- setColorHex('');
- setColorInput('');
- }}
- className="px-2 py-1 text-xs text-bambu-gray hover:text-white bg-bambu-dark-tertiary rounded"
- title={t('configureAmsSlot.clearCustomColor')}
- >
- {t('configureAmsSlot.clear')}
- </button>
- )}
- </div>
- {colorHex && (
- <p className="text-xs text-bambu-gray mt-1.5">
- {t('configureAmsSlot.hexLabel', { hex: colorHex })}
- </p>
- )}
- </div>
- </>
- )}
- </div>
- {/* Footer */}
- <div className="flex justify-between p-4 border-t border-bambu-dark-tertiary shrink-0">
- {/* Reset button on the left */}
- <Button
- variant="secondary"
- onClick={() => resetMutation.mutate()}
- disabled={resetMutation.isPending || configureMutation.isPending}
- className="text-red-400 hover:text-red-300 hover:bg-red-500/10"
- >
- {resetMutation.isPending ? (
- <>
- <Loader2 className="w-4 h-4 animate-spin" />
- {t('configureAmsSlot.resetting')}
- </>
- ) : (
- <>
- <RotateCcw className="w-4 h-4" />
- {t('configureAmsSlot.resetSlot')}
- </>
- )}
- </Button>
- {/* Cancel and Configure buttons on the right */}
- <div className="flex gap-2">
- <Button variant="secondary" onClick={onClose}>
- {t('configureAmsSlot.cancel')}
- </Button>
- <Button
- onClick={() => configureMutation.mutate()}
- disabled={!canSave}
- >
- {configureMutation.isPending ? (
- <>
- <Loader2 className="w-4 h-4 animate-spin" />
- {t('configureAmsSlot.configuring')}
- </>
- ) : (
- <>
- <Settings2 className="w-4 h-4" />
- {t('configureAmsSlot.configureSlot')}
- </>
- )}
- </Button>
- </div>
- </div>
- {/* Error */}
- {(configureMutation.isError || resetMutation.isError) && (
- <div className="mx-4 mb-4 p-2 bg-red-500/20 border border-red-500/50 rounded text-sm text-red-400">
- {(configureMutation.error as Error)?.message || (resetMutation.error as Error)?.message}
- </div>
- )}
- </div>
- </div>
- );
- }
|