| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567 |
- import React, { useState, useEffect, useCallback } from 'react';
- import { useQuery, useMutation } from '@tanstack/react-query';
- import { useTranslation } from 'react-i18next';
- import {
- Gauge,
- Loader2,
- RefreshCw,
- Printer,
- Plus,
- X,
- AlertCircle,
- WifiOff,
- Trash2,
- Search,
- Copy,
- Download,
- Upload,
- CheckSquare,
- Square,
- StickyNote,
- } from 'lucide-react';
- import { api } from '../api/client';
- import type { KProfile, KProfileCreate, KProfileDelete, Permission } from '../api/client';
- import { Card, CardContent } from './Card';
- import { Button } from './Button';
- import { useToast } from '../contexts/ToastContext';
- import { useAuth } from '../contexts/AuthContext';
- interface KProfileCardProps {
- profile: KProfile;
- onEdit: () => void;
- onCopy?: () => void;
- selectionMode?: boolean;
- isSelected?: boolean;
- onToggleSelect?: () => void;
- note?: string; // Note text to display as preview
- }
- // Truncate to 3 decimal places (like Bambu Studio) instead of rounding
- const truncateK = (value: string) => {
- const num = parseFloat(value);
- return (Math.trunc(num * 1000) / 1000).toFixed(3);
- };
- // Get flow type label from nozzle_id (e.g., "HH00-0.4" -> "HF", "HS00-0.4" -> "S")
- const getFlowTypeLabel = (nozzleId: string) => {
- if (nozzleId.startsWith('HH')) return 'HF'; // High Flow
- return 'S'; // Standard Flow (default)
- };
- // Extract nozzle type prefix from nozzle_id (e.g., "HH00-0.4" -> "HH00")
- const getNozzleTypePrefix = (nozzleId: string) => {
- const match = nozzleId.match(/^([A-Z]{2}\d{2})/);
- return match ? match[1] : 'HH00';
- };
- // Extract filament name from profile name (e.g., "High Flow_Devil Design PLA Basic" -> "Devil Design PLA Basic")
- const extractFilamentName = (profileName: string) => {
- // Profile names are formatted as "{Flow Type}_{Filament Name}" or "{Flow Type} {Filament Name}"
- // Remove common prefixes - check both underscore and space separators
- const prefixes = [
- 'High Flow_', 'High Flow ', // underscore or space
- 'Standard_', 'Standard ',
- 'HF_', 'HF ',
- 'S_', 'S ',
- ];
- for (const prefix of prefixes) {
- if (profileName.startsWith(prefix)) {
- return profileName.slice(prefix.length);
- }
- }
- // If no prefix found, check for underscore separator
- const underscoreIdx = profileName.indexOf('_');
- if (underscoreIdx > 0) {
- return profileName.slice(underscoreIdx + 1);
- }
- return profileName;
- };
- function KProfileCard({ profile, onEdit, onCopy, selectionMode, isSelected, onToggleSelect, note }: KProfileCardProps) {
- const flowType = getFlowTypeLabel(profile.nozzle_id);
- const diameter = profile.nozzle_diameter;
- const handleClick = () => {
- if (selectionMode && onToggleSelect) {
- onToggleSelect();
- } else {
- onEdit();
- }
- };
- return (
- <div className="flex items-center gap-2">
- {selectionMode && (
- <button
- onClick={onToggleSelect}
- className="text-bambu-gray hover:text-white transition-colors p-1"
- >
- {isSelected ? (
- <CheckSquare className="w-4 h-4 text-bambu-green" />
- ) : (
- <Square className="w-4 h-4" />
- )}
- </button>
- )}
- <button
- onClick={handleClick}
- className={`flex-1 text-left px-3 py-2 bg-bambu-dark rounded hover:bg-bambu-dark-tertiary transition-colors ${isSelected ? 'ring-1 ring-bambu-green' : ''}`}
- >
- <div className="flex items-center gap-2">
- <span className="text-bambu-green font-mono text-sm font-bold whitespace-nowrap">
- {truncateK(profile.k_value)}
- </span>
- <span className="text-white text-sm truncate flex-1" title={profile.name}>
- {profile.name || 'Unnamed'}
- </span>
- {note && (
- <span title="Has note">
- <StickyNote className="w-3 h-3 text-yellow-500" />
- </span>
- )}
- <span className="text-xs text-bambu-gray whitespace-nowrap">
- {flowType} {diameter}
- </span>
- </div>
- {note && (
- <div className="text-xs mt-0.5 truncate text-yellow-500/70" title={note}>
- Note: {note.length > 50 ? note.substring(0, 50) + '...' : note}
- </div>
- )}
- </button>
- {!selectionMode && onCopy && (
- <button
- onClick={(e) => {
- e.stopPropagation();
- onCopy();
- }}
- className="text-bambu-gray hover:text-white transition-colors p-1"
- title="Copy profile"
- >
- <Copy className="w-4 h-4" />
- </button>
- )}
- </div>
- );
- }
- interface KProfileModalProps {
- profile?: KProfile;
- printerId: number;
- nozzleDiameter: string;
- existingProfiles?: KProfile[]; // Existing profiles for filament selection
- builtinFilaments?: { filament_id: string; name: string }[]; // Filament ID → name lookup
- isDualNozzle?: boolean; // Whether this is a dual-nozzle printer
- initialNote?: string; // Initial note value for the profile
- initialNoteKey?: string | null; // Key the note was stored under (for clearing)
- onClose: () => void;
- onSave: () => void;
- onSaveNote?: (settingId: string, note: string) => void; // Callback to save note
- hasPermission: (permission: Permission) => boolean;
- }
- function KProfileModal({
- profile,
- printerId,
- nozzleDiameter,
- existingProfiles = [],
- builtinFilaments = [],
- isDualNozzle = false,
- initialNote = '',
- initialNoteKey = null,
- onClose,
- onSave,
- onSaveNote,
- hasPermission,
- }: KProfileModalProps) {
- const { t } = useTranslation();
- const { showToast } = useToast();
- const [name, setName] = useState(profile?.name || '');
- const [kValue, setKValue] = useState(
- profile?.k_value ? truncateK(profile.k_value) : '0.020'
- );
- const [filamentId, setFilamentId] = useState(profile?.filament_id || '');
- // Split nozzle into type and diameter
- const [nozzleType, setNozzleType] = useState(
- profile?.nozzle_id ? getNozzleTypePrefix(profile.nozzle_id) : 'HH00'
- );
- const [modalDiameter, setModalDiameter] = useState(
- profile?.nozzle_diameter || nozzleDiameter
- );
- // For new profiles on dual-nozzle: allow selecting multiple extruders
- // For editing: use single extruder from the profile
- const [selectedExtruders, setSelectedExtruders] = useState<number[]>(
- profile ? [profile.extruder_id] : isDualNozzle ? [0, 1] : [0] // Default: both extruders for new dual-nozzle profiles
- );
- const [isSyncing, setIsSyncing] = useState(false);
- const [savingProgress, setSavingProgress] = useState({ current: 0, total: 0 });
- const [note, setNote] = useState(initialNote);
- // Extract unique filaments from existing K-profiles on the printer
- // Use builtin filament table for accurate name resolution (filament_id → name)
- // Falls back to extracting from profile name for custom/unknown presets
- const knownFilaments = React.useMemo(() => {
- // Build lookup map from builtin filament names (includes cloud presets from parent)
- const builtinMap = new Map<string, string>();
- for (const bf of builtinFilaments) {
- builtinMap.set(bf.filament_id, bf.name);
- }
- const filamentMap = new Map<string, { id: string; name: string }>();
- for (const p of existingProfiles) {
- if (p.filament_id && !filamentMap.has(p.filament_id)) {
- // Prefer builtin name (accurate), fall back to extracting from profile name
- const builtinName = builtinMap.get(p.filament_id);
- const filamentName = builtinName || extractFilamentName(p.name || '');
- filamentMap.set(p.filament_id, {
- id: p.filament_id,
- name: filamentName || p.filament_id,
- });
- }
- }
- return Array.from(filamentMap.values()).sort((a, b) =>
- a.name.localeCompare(b.name)
- );
- }, [existingProfiles, builtinFilaments]);
- const saveMutation = useMutation({
- mutationFn: (data: KProfileCreate) => {
- console.log('[KProfile] Calling API...');
- return api.setKProfile(printerId, data);
- },
- onSuccess: (result) => {
- console.log('[KProfile] Save success:', result);
- showToast(t('kProfiles.toast.profileSaved'));
- // Save note if it changed (including clearing it)
- if (onSaveNote && note !== initialNote) {
- let profileKey: string;
- if (note === '' && initialNoteKey) {
- // Clearing note: use the same key it was stored under
- profileKey = initialNoteKey;
- } else if (profile && profile.slot_id > 0) {
- // Editing: use setting_id if available, or composite key with slot_id
- profileKey = profile.setting_id || `slot_${profile.slot_id}_${profile.filament_id}_${profile.extruder_id}`;
- } else {
- // New profile: use name as key (will be matched when profile is loaded)
- profileKey = `name_${name}_${filamentId}`;
- }
- onSaveNote(profileKey, note);
- }
- // Show syncing indicator while printer processes the command
- setIsSyncing(true);
- // Add delay before closing to give printer time to process the save
- // onSave will trigger refetch in the parent component
- setTimeout(() => {
- setIsSyncing(false);
- onSave();
- }, 2500);
- },
- onError: (error: Error) => {
- console.error('[KProfile] Save error:', error);
- showToast(error.message, 'error');
- setIsSyncing(false);
- },
- });
- const deleteMutation = useMutation({
- mutationFn: (data: KProfileDelete) => {
- console.log('[KProfile] Deleting profile...');
- return api.deleteKProfile(printerId, data);
- },
- onSuccess: (result) => {
- console.log('[KProfile] Delete success:', result);
- showToast(t('kProfiles.toast.profileDeleted'));
- // Show syncing indicator while printer processes the command
- setIsSyncing(true);
- // Add longer delay for delete - printer needs more time to process
- // before it can return the updated profile list
- setTimeout(() => {
- setIsSyncing(false);
- onClose();
- }, 4000);
- },
- onError: (error: Error) => {
- console.error('[KProfile] Delete error:', error);
- showToast(error.message, 'error');
- setIsSyncing(false);
- },
- });
- const [showDeleteConfirm, setShowDeleteConfirm] = useState(false);
- const handleDelete = () => {
- if (!profile) return;
- deleteMutation.mutate({
- slot_id: profile.slot_id,
- extruder_id: profile.extruder_id,
- nozzle_id: profile.nozzle_id,
- nozzle_diameter: profile.nozzle_diameter,
- filament_id: profile.filament_id,
- setting_id: profile.setting_id,
- });
- };
- const handleSubmit = async (e: React.FormEvent) => {
- e.preventDefault();
- // Validate at least one extruder is selected for dual-nozzle
- if (isDualNozzle && !profile && selectedExtruders.length === 0) {
- showToast(t('kProfiles.toast.selectAtLeastOneExtruder'), 'error');
- return;
- }
- // Format k_value to 6 decimal places for Bambu protocol
- const formattedKValue = parseFloat(kValue).toFixed(6);
- // Combine nozzle type and diameter into nozzle_id (e.g., "HH00-0.4")
- const nozzleId = `${nozzleType}-${modalDiameter}`;
- // For editing or single extruder: just save one profile
- if (profile || selectedExtruders.length === 1) {
- const payload = {
- name: name,
- k_value: formattedKValue,
- filament_id: filamentId,
- nozzle_id: nozzleId,
- nozzle_diameter: modalDiameter,
- extruder_id: profile ? profile.extruder_id : selectedExtruders[0],
- setting_id: profile?.setting_id,
- slot_id: profile?.slot_id ?? 0,
- };
- console.log('[KProfile] Saving profile:', payload);
- saveMutation.mutate(payload);
- return;
- }
- // For new profiles with multiple extruders: use batch endpoint
- setIsSyncing(true);
- setSavingProgress({ current: 1, total: selectedExtruders.length });
- // Build payload for all selected extruders
- const batchPayload = selectedExtruders.map(extruderId => ({
- name: name,
- k_value: formattedKValue,
- filament_id: filamentId,
- nozzle_id: nozzleId,
- nozzle_diameter: modalDiameter,
- extruder_id: extruderId,
- setting_id: undefined,
- slot_id: 0,
- }));
- console.log(`[KProfile] Saving ${batchPayload.length} profiles in batch:`, batchPayload);
- try {
- await api.setKProfilesBatch(printerId, batchPayload);
- showToast(t('kProfiles.toast.profilesSaved', { count: selectedExtruders.length }));
- // Save note for new batch profiles
- if (onSaveNote && note) {
- const profileKey = `name_${name}_${filamentId}`;
- onSaveNote(profileKey, note);
- }
- } catch (error) {
- console.error('[KProfile] Failed to save batch:', error);
- showToast(t('kProfiles.toast.failedToSaveBatch'), 'error');
- setIsSyncing(false);
- setSavingProgress({ current: 0, total: 0 });
- return;
- }
- setSavingProgress({ current: selectedExtruders.length, total: selectedExtruders.length });
- // Wait for final sync before closing
- // onSave will trigger refetch in the parent component
- setTimeout(() => {
- setIsSyncing(false);
- setSavingProgress({ current: 0, total: 0 });
- onSave();
- }, 3000);
- };
- return (
- <div className="fixed inset-0 bg-black/50 flex items-center justify-center z-50 p-4">
- <Card className="w-full max-w-md relative">
- {/* Syncing overlay */}
- {isSyncing && (
- <div className="absolute inset-0 bg-bambu-dark-secondary/90 flex flex-col items-center justify-center z-10 rounded-lg">
- <Loader2 className="w-8 h-8 text-bambu-green animate-spin mb-3" />
- <p className="text-white font-medium">
- {savingProgress.total > 1
- ? t('kProfiles.modal.savingExtruder', { current: savingProgress.current, total: savingProgress.total })
- : t('kProfiles.modal.syncing')}
- </p>
- <p className="text-bambu-gray text-sm mt-1">{t('kProfiles.modal.pleaseWait')}</p>
- </div>
- )}
- <CardContent className="p-0">
- <div className="flex items-center justify-between p-4 border-b border-bambu-dark-tertiary">
- <h2 className="text-xl font-semibold text-white">
- {profile ? t('kProfiles.modal.editTitle') : t('kProfiles.modal.addTitle')}
- </h2>
- <button
- onClick={onClose}
- className="text-bambu-gray hover:text-white transition-colors"
- disabled={isSyncing}
- >
- <X className="w-5 h-5" />
- </button>
- </div>
- <form onSubmit={handleSubmit} className="p-4 space-y-4">
- {/* Profile Name - read-only when editing */}
- <div>
- <label className="block text-sm text-bambu-gray mb-1">{t('kProfiles.modal.profileName')}</label>
- <input
- type="text"
- value={name}
- onChange={(e) => setName(e.target.value)}
- disabled={!!profile}
- 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 ${profile ? 'opacity-60 cursor-not-allowed' : ''}`}
- placeholder={t('kProfiles.modal.profileNamePlaceholder')}
- required={!profile}
- />
- </div>
- {/* K-Value - always editable */}
- <div>
- <label className="block text-sm text-bambu-gray mb-1">{t('kProfiles.modal.kValue')}</label>
- <input
- type="text"
- inputMode="decimal"
- value={kValue}
- onChange={(e) => {
- // Allow typing any decimal value
- const val = e.target.value;
- if (val === '' || /^\d*\.?\d*$/.test(val)) {
- setKValue(val);
- }
- }}
- onBlur={(e) => {
- // Format to 3 decimal places on blur
- const num = parseFloat(e.target.value);
- if (!isNaN(num)) {
- setKValue((Math.trunc(num * 1000) / 1000).toFixed(3));
- }
- }}
- 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 font-mono"
- placeholder={t('kProfiles.modal.kValuePlaceholder')}
- required
- />
- <p className="text-xs text-bambu-gray mt-1">
- {t('kProfiles.modal.kValueHelp')}
- </p>
- </div>
- {/* Filament - read-only when editing */}
- <div>
- <label className="block text-sm text-bambu-gray mb-1">{t('kProfiles.modal.filament')}</label>
- <select
- value={filamentId}
- onChange={(e) => {
- const newFilamentId = e.target.value;
- setFilamentId(newFilamentId);
- // Auto-generate profile name when filament is selected (for new profiles)
- // Only auto-generate if name is empty - don't overwrite user input
- if (!profile && newFilamentId && !name) {
- const selectedFilament = knownFilaments.find(f => f.id === newFilamentId);
- if (selectedFilament) {
- const flowLabel = nozzleType === 'HH00' ? 'HF' : 'S';
- setName(`${flowLabel} ${selectedFilament.name}`);
- }
- }
- }}
- disabled={!!profile}
- 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 ${profile ? 'opacity-60 cursor-not-allowed' : ''}`}
- required={!profile}
- >
- <option value="">{t('kProfiles.modal.selectFilament')}</option>
- {/* Show current filament when editing - look up from knownFilaments */}
- {profile?.filament_id && (
- <option key={profile.filament_id} value={profile.filament_id}>
- {knownFilaments.find(f => f.id === profile.filament_id)?.name || profile.filament_id}
- </option>
- )}
- {/* Show known filaments from existing K-profiles (for new profiles) */}
- {!profile && knownFilaments.map((f) => (
- <option key={f.id} value={f.id}>
- {f.name}
- </option>
- ))}
- </select>
- {!profile && knownFilaments.length === 0 && (
- <p className="text-xs text-bambu-gray mt-1">
- {t('kProfiles.modal.noFilamentsHelp')}
- </p>
- )}
- </div>
- {/* Flow Type and Nozzle Size - read-only when editing */}
- <div className="grid grid-cols-2 gap-4">
- <div>
- <label className="block text-sm text-bambu-gray mb-1">{t('kProfiles.modal.flowType')}</label>
- <select
- value={nozzleType}
- onChange={(e) => {
- const newNozzleType = e.target.value;
- setNozzleType(newNozzleType);
- // Update profile name when flow type changes (for new profiles)
- // Only auto-generate if name is empty - don't overwrite user input
- if (!profile && filamentId && !name) {
- const selectedFilament = knownFilaments.find(f => f.id === filamentId);
- if (selectedFilament) {
- const flowLabel = newNozzleType === 'HS00' ? 'HF' : 'S';
- setName(`${flowLabel} ${selectedFilament.name}`);
- }
- }
- }}
- disabled={!!profile}
- 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 ${profile ? 'opacity-60 cursor-not-allowed' : ''}`}
- >
- <option value="HH00">{t('kProfiles.modal.highFlow')}</option>
- <option value="HS00">{t('kProfiles.modal.standard')}</option>
- </select>
- </div>
- <div>
- <label className="block text-sm text-bambu-gray mb-1">{t('kProfiles.modal.nozzleSize')}</label>
- <select
- value={modalDiameter}
- onChange={(e) => setModalDiameter(e.target.value)}
- disabled={!!profile}
- 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 ${profile ? 'opacity-60 cursor-not-allowed' : ''}`}
- >
- <option value="0.2">0.2mm</option>
- <option value="0.4">0.4mm</option>
- <option value="0.6">0.6mm</option>
- <option value="0.8">0.8mm</option>
- </select>
- </div>
- </div>
- {/* Extruder - only show for dual-nozzle printers */}
- {isDualNozzle && (
- <div>
- <label className="block text-sm text-bambu-gray mb-1">
- {profile ? t('kProfiles.modal.extruder') : t('kProfiles.modal.extruders')}
- </label>
- {profile ? (
- // Read-only display for editing
- <div className="px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white opacity-60">
- {profile.extruder_id === 1 ? t('kProfiles.modal.left') : t('kProfiles.modal.right')}
- </div>
- ) : (
- // Checkboxes for new profile - can select both
- <div className="flex gap-4">
- <label className="flex items-center gap-2 cursor-pointer">
- <input
- type="checkbox"
- checked={selectedExtruders.includes(1)}
- onChange={(e) => {
- if (e.target.checked) {
- setSelectedExtruders([...selectedExtruders, 1]);
- } else {
- setSelectedExtruders(selectedExtruders.filter(id => id !== 1));
- }
- }}
- className="w-4 h-4 rounded border-bambu-dark-tertiary bg-bambu-dark text-bambu-green focus:ring-bambu-green focus:ring-offset-0 accent-bambu-green"
- />
- <span className="text-white">{t('kProfiles.modal.left')}</span>
- </label>
- <label className="flex items-center gap-2 cursor-pointer">
- <input
- type="checkbox"
- checked={selectedExtruders.includes(0)}
- onChange={(e) => {
- if (e.target.checked) {
- setSelectedExtruders([...selectedExtruders, 0]);
- } else {
- setSelectedExtruders(selectedExtruders.filter(id => id !== 0));
- }
- }}
- className="w-4 h-4 rounded border-bambu-dark-tertiary bg-bambu-dark text-bambu-green focus:ring-bambu-green focus:ring-offset-0 accent-bambu-green"
- />
- <span className="text-white">{t('kProfiles.modal.right')}</span>
- </label>
- </div>
- )}
- </div>
- )}
- {/* Notes */}
- <div>
- <label className="block text-sm text-bambu-gray mb-1">{t('kProfiles.modal.notes')}</label>
- <textarea
- value={note}
- onChange={(e) => setNote(e.target.value)}
- placeholder={t('kProfiles.modal.notesPlaceholder')}
- rows={2}
- className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white placeholder-bambu-gray focus:border-bambu-green focus:outline-none resize-none"
- />
- <p className="text-xs text-bambu-gray mt-1">
- {t('kProfiles.modal.notesHelp')}
- </p>
- </div>
- <div className="flex gap-2 pt-4">
- {profile && (
- <Button
- type="button"
- variant="secondary"
- onClick={() => setShowDeleteConfirm(true)}
- disabled={deleteMutation.isPending || isSyncing || !hasPermission('kprofiles:delete')}
- title={!hasPermission('kprofiles:delete') ? t('kProfiles.permission.noDelete') : undefined}
- className="text-red-500 hover:bg-red-500/10"
- >
- {deleteMutation.isPending ? (
- <Loader2 className="w-4 h-4 animate-spin" />
- ) : (
- <Trash2 className="w-4 h-4" />
- )}
- </Button>
- )}
- <Button
- type="button"
- variant="secondary"
- onClick={onClose}
- disabled={isSyncing}
- className="flex-1"
- >
- {t('common.cancel')}
- </Button>
- <Button
- type="submit"
- disabled={saveMutation.isPending || isSyncing || !hasPermission(profile ? 'kprofiles:update' : 'kprofiles:create')}
- title={!hasPermission(profile ? 'kprofiles:update' : 'kprofiles:create') ? t(profile ? 'kProfiles.permission.noUpdate' : 'kProfiles.permission.noCreate') : undefined}
- className="flex-1"
- >
- {saveMutation.isPending ? (
- <Loader2 className="w-4 h-4 animate-spin" />
- ) : (
- <Gauge className="w-4 h-4" />
- )}
- {t('common.save')}
- </Button>
- </div>
- </form>
- </CardContent>
- </Card>
- {/* Delete Confirmation Modal */}
- {showDeleteConfirm && (
- <div className="fixed inset-0 bg-black/70 flex items-center justify-center z-[60]">
- <Card className="w-full max-w-sm">
- <CardContent className="p-6">
- <div className="flex items-center gap-3 mb-4">
- <div className="w-10 h-10 rounded-full bg-red-500/20 flex items-center justify-center">
- <Trash2 className="w-5 h-5 text-red-500" />
- </div>
- <div>
- <h3 className="text-lg font-semibold text-white">{t('kProfiles.deleteConfirm.title')}</h3>
- <p className="text-sm text-bambu-gray">{t('kProfiles.deleteConfirm.cannotUndo')}</p>
- </div>
- </div>
- <p className="text-bambu-gray mb-6">
- {t('kProfiles.deleteConfirm.message', { name: profile?.name })}
- </p>
- <div className="flex gap-3">
- <Button
- variant="secondary"
- onClick={() => setShowDeleteConfirm(false)}
- className="flex-1"
- >
- {t('common.cancel')}
- </Button>
- <Button
- onClick={() => {
- setShowDeleteConfirm(false);
- handleDelete();
- }}
- disabled={deleteMutation.isPending}
- className="flex-1 bg-red-500 hover:bg-red-600 text-white"
- >
- {deleteMutation.isPending ? (
- <Loader2 className="w-4 h-4 animate-spin" />
- ) : (
- <Trash2 className="w-4 h-4" />
- )}
- {t('common.delete')}
- </Button>
- </div>
- </CardContent>
- </Card>
- </div>
- )}
- </div>
- );
- }
- type ExtruderFilter = 'all' | 'left' | 'right';
- type FlowTypeFilter = 'all' | 'hf' | 's';
- type SortOption = 'name' | 'k_value' | 'filament';
- // localStorage keys
- const STORAGE_KEYS = {
- NOZZLE_DIAMETER: 'bambusy_kprofiles_nozzle',
- SORT_OPTION: 'bambusy_kprofiles_sort',
- };
- export function KProfilesView() {
- const { t } = useTranslation();
- const { showToast } = useToast();
- const { hasPermission } = useAuth();
- const [selectedPrinter, setSelectedPrinter] = useState<number | null>(null);
- // Load nozzle diameter from localStorage
- const [nozzleDiameter, setNozzleDiameter] = useState(() => {
- const saved = localStorage.getItem(STORAGE_KEYS.NOZZLE_DIAMETER);
- return saved || '0.4';
- });
- const [editingProfile, setEditingProfile] = useState<KProfile | null>(null);
- const [showAddModal, setShowAddModal] = useState(false);
- const [copyingProfile, setCopyingProfile] = useState<KProfile | null>(null);
- const [searchQuery, setSearchQuery] = useState('');
- const [extruderFilter, setExtruderFilter] = useState<ExtruderFilter>('all');
- const [flowTypeFilter, setFlowTypeFilter] = useState<FlowTypeFilter>('all');
- // Load sort option from localStorage
- const [sortOption, setSortOption] = useState<SortOption>(() => {
- const saved = localStorage.getItem(STORAGE_KEYS.SORT_OPTION);
- return (saved as SortOption) || 'name';
- });
- // Bulk selection mode
- // Use composite key: `${slot_id}_${extruder_id}` since slot_id alone is not unique across extruders
- const [selectionMode, setSelectionMode] = useState(false);
- const [selectedProfiles, setSelectedProfiles] = useState<Set<string>>(new Set());
- const [showBulkDeleteConfirm, setShowBulkDeleteConfirm] = useState(false);
- const [bulkDeleteInProgress, setBulkDeleteInProgress] = useState(false);
- // Helper to create unique profile key for selection - wrapped in useCallback to prevent re-renders
- const getProfileKey = useCallback((profile: KProfile) => `${profile.slot_id}_${profile.extruder_id}`, []);
- // Save nozzle diameter to localStorage when it changes
- useEffect(() => {
- localStorage.setItem(STORAGE_KEYS.NOZZLE_DIAMETER, nozzleDiameter);
- }, [nozzleDiameter]);
- // Save sort option to localStorage when it changes
- useEffect(() => {
- localStorage.setItem(STORAGE_KEYS.SORT_OPTION, sortOption);
- }, [sortOption]);
- // Get available printers
- const { data: printers, isLoading: printersLoading } = useQuery({
- queryKey: ['printers'],
- queryFn: api.getPrinters,
- });
- // Get K-profiles for selected printer (filtered by nozzle diameter)
- const {
- data: kprofiles,
- isLoading: kprofilesLoading,
- isFetching,
- error: kprofilesError,
- refetch: refetchProfiles,
- } = useQuery({
- queryKey: ['kprofiles', selectedPrinter, nozzleDiameter],
- queryFn: async () => {
- console.log('[KProfiles] Fetching profiles for printer', selectedPrinter, 'nozzle', nozzleDiameter);
- const result = await api.getKProfiles(selectedPrinter!, nozzleDiameter);
- console.log('[KProfiles] Received profiles:', result?.profiles?.length || 0, 'profiles');
- return result;
- },
- enabled: !!selectedPrinter,
- retry: false,
- staleTime: 0, // Always consider data stale to ensure fresh fetch
- gcTime: 0, // Don't cache results
- refetchOnMount: 'always', // Always refetch when component mounts
- });
- // Also fetch 0.4mm profiles for the filament dropdown (most filaments are calibrated for 0.4mm)
- const { data: allProfiles } = useQuery({
- queryKey: ['kprofiles', selectedPrinter, '0.4'],
- queryFn: () => api.getKProfiles(selectedPrinter!, '0.4'),
- enabled: !!selectedPrinter,
- staleTime: 60000, // Cache for 1 minute
- });
- // Fetch builtin filament names for accurate filament_id → name resolution
- const { data: builtinFilaments } = useQuery({
- queryKey: ['builtinFilaments'],
- queryFn: () => api.getBuiltinFilaments(),
- staleTime: 300000, // Cache for 5 minutes (static data)
- });
- // Fetch filament_id → name mapping for user cloud presets (P* IDs)
- const { data: filamentIdMap } = useQuery({
- queryKey: ['filamentIdMap'],
- queryFn: () => api.getFilamentIdMap(),
- staleTime: 300000, // Cache for 5 minutes
- });
- // Fetch K-profile notes (stored locally)
- const {
- data: notesData,
- refetch: refetchNotes,
- } = useQuery({
- queryKey: ['kprofile-notes', selectedPrinter],
- queryFn: () => api.getKProfileNotes(selectedPrinter!),
- enabled: !!selectedPrinter,
- staleTime: 30000, // Cache for 30 seconds
- });
- // Check if error is due to printer not being connected
- const isOfflineError = kprofilesError?.message?.includes('not connected');
- // Auto-select first connected printer
- useEffect(() => {
- if (!selectedPrinter && printers && printers.length > 0) {
- const activePrinter = printers.find((p) => p.is_active);
- if (activePrinter) {
- setSelectedPrinter(activePrinter.id);
- }
- }
- }, [selectedPrinter, printers]);
- // Refetch profiles when printer selection changes
- useEffect(() => {
- if (selectedPrinter) {
- // Delay refetch to ensure query is enabled after state update
- const timer = setTimeout(() => {
- refetchProfiles();
- }, 150);
- return () => clearTimeout(timer);
- }
- }, [selectedPrinter, nozzleDiameter]); // eslint-disable-line react-hooks/exhaustive-deps
- // Get connected printers for display
- const connectedPrinters = printers?.filter((p) => p.is_active) || [];
- // Build filament lookup for name resolution (builtin + user cloud presets)
- const builtinFilamentMap = React.useMemo(() => {
- const map = new Map<string, string>();
- if (builtinFilaments) {
- for (const bf of builtinFilaments) {
- map.set(bf.filament_id, bf.name);
- }
- }
- // Also add user cloud presets (P* filament_ids resolved from cloud details)
- if (filamentIdMap) {
- for (const [fid, name] of Object.entries(filamentIdMap)) {
- if (!map.has(fid)) {
- map.set(fid, name);
- }
- }
- }
- return map;
- }, [builtinFilaments, filamentIdMap]);
- // Enriched builtin filaments array (builtin + cloud presets merged)
- // Pass this to modals so they have the full filament name lookup
- const enrichedBuiltinFilaments = React.useMemo(() => {
- return Array.from(builtinFilamentMap.entries()).map(([fid, name]) => ({
- filament_id: fid,
- name,
- }));
- }, [builtinFilamentMap]);
- // Resolve filament name: builtin table first, then extract from profile name
- const resolveFilamentName = React.useCallback((profile: KProfile) => {
- return builtinFilamentMap.get(profile.filament_id) || extractFilamentName(profile.name);
- }, [builtinFilamentMap]);
- // Filter and sort profiles
- // Note: nozzle diameter filtering is done server-side via MQTT request
- const filteredProfiles = React.useMemo(() => {
- if (!kprofiles?.profiles) return [];
- const filtered = kprofiles.profiles.filter((p) => {
- // Search filter - match name or filament_id (case-insensitive)
- const query = searchQuery.toLowerCase();
- const matchesSearch =
- !query ||
- p.name.toLowerCase().includes(query) ||
- p.filament_id.toLowerCase().includes(query);
- // Extruder filter
- const matchesExtruder =
- extruderFilter === 'all' ||
- (extruderFilter === 'left' && p.extruder_id === 1) ||
- (extruderFilter === 'right' && p.extruder_id === 0);
- // Flow type filter (HH = High Flow, HS = Standard)
- const matchesFlowType =
- flowTypeFilter === 'all' ||
- (flowTypeFilter === 'hf' && p.nozzle_id.startsWith('HH')) ||
- (flowTypeFilter === 's' && p.nozzle_id.startsWith('HS'));
- return matchesSearch && matchesExtruder && matchesFlowType;
- });
- // Sort profiles
- return filtered.sort((a, b) => {
- switch (sortOption) {
- case 'k_value':
- return parseFloat(a.k_value) - parseFloat(b.k_value);
- case 'filament':
- return resolveFilamentName(a).localeCompare(resolveFilamentName(b));
- case 'name':
- default:
- return a.name.localeCompare(b.name);
- }
- });
- }, [kprofiles?.profiles, searchQuery, extruderFilter, flowTypeFilter, sortOption, resolveFilamentName]);
- // Check if selected printer is dual-nozzle (auto-detected from MQTT temperature data)
- const selectedPrinterData = printers?.find((p) => p.id === selectedPrinter);
- const isDualNozzle = selectedPrinterData?.nozzle_count === 2;
- // Keyboard shortcuts
- useEffect(() => {
- const handleKeyDown = (e: KeyboardEvent) => {
- // Don't trigger shortcuts when typing in input fields
- if (e.target instanceof HTMLInputElement || e.target instanceof HTMLTextAreaElement || e.target instanceof HTMLSelectElement) {
- return;
- }
- // Don't trigger when modal is open
- if (editingProfile || showAddModal || copyingProfile) {
- return;
- }
- if (e.key === 'r' || e.key === 'R') {
- e.preventDefault();
- refetchProfiles();
- } else if (e.key === 'n' || e.key === 'N') {
- e.preventDefault();
- setShowAddModal(true);
- } else if (e.key === 'Escape' && selectionMode) {
- e.preventDefault();
- setSelectionMode(false);
- setSelectedProfiles(new Set());
- }
- };
- window.addEventListener('keydown', handleKeyDown);
- return () => window.removeEventListener('keydown', handleKeyDown);
- }, [editingProfile, showAddModal, copyingProfile, selectionMode, refetchProfiles]);
- // Export profiles to JSON file
- const handleExport = useCallback(() => {
- if (!kprofiles?.profiles || kprofiles.profiles.length === 0) {
- showToast(t('kProfiles.toast.noProfilesToExport'), 'error');
- return;
- }
- const exportData = {
- version: 1,
- exported_at: new Date().toISOString(),
- printer: selectedPrinterData?.name || 'Unknown',
- nozzle_diameter: nozzleDiameter,
- profiles: kprofiles.profiles.map(p => ({
- name: p.name,
- k_value: p.k_value,
- filament_id: p.filament_id,
- nozzle_id: p.nozzle_id,
- nozzle_diameter: p.nozzle_diameter,
- extruder_id: p.extruder_id,
- })),
- };
- const blob = new Blob([JSON.stringify(exportData, null, 2)], { type: 'application/json' });
- const url = URL.createObjectURL(blob);
- const a = document.createElement('a');
- a.href = url;
- a.download = `kprofiles_${selectedPrinterData?.name || 'printer'}_${nozzleDiameter}mm_${new Date().toISOString().split('T')[0]}.json`;
- document.body.appendChild(a);
- a.click();
- document.body.removeChild(a);
- URL.revokeObjectURL(url);
- showToast(t('kProfiles.toast.exportedProfiles', { count: kprofiles.profiles.length }));
- }, [kprofiles?.profiles, selectedPrinterData, nozzleDiameter, showToast, t]);
- // Import profiles from JSON file
- const handleImport = useCallback(() => {
- const input = document.createElement('input');
- input.type = 'file';
- input.accept = '.json';
- input.onchange = async (e) => {
- const file = (e.target as HTMLInputElement).files?.[0];
- if (!file) return;
- try {
- const text = await file.text();
- const data = JSON.parse(text);
- if (!data.profiles || !Array.isArray(data.profiles)) {
- showToast(t('kProfiles.toast.invalidFileFormat'), 'error');
- return;
- }
- // Import profiles one by one
- let imported = 0;
- for (const p of data.profiles) {
- if (!p.name || !p.k_value || !p.filament_id) continue;
- try {
- await api.setKProfile(selectedPrinter!, {
- name: p.name,
- k_value: parseFloat(p.k_value).toFixed(6),
- filament_id: p.filament_id,
- nozzle_id: p.nozzle_id || `HH00-${nozzleDiameter}`,
- nozzle_diameter: p.nozzle_diameter || nozzleDiameter,
- extruder_id: p.extruder_id ?? 0,
- slot_id: 0, // Always create new
- });
- imported++;
- // Small delay between imports
- await new Promise(resolve => setTimeout(resolve, 500));
- } catch (err) {
- console.error('Failed to import profile:', p.name, err);
- }
- }
- showToast(t('kProfiles.toast.importedProfiles', { count: imported, total: data.profiles.length }));
- refetchProfiles();
- } catch (err) {
- console.error('Import error:', err);
- showToast(t('kProfiles.toast.failedToParseImport'), 'error');
- }
- };
- input.click();
- }, [selectedPrinter, nozzleDiameter, showToast, refetchProfiles, t]);
- // Toggle profile selection using composite key
- const toggleProfileSelection = useCallback((profileKey: string) => {
- setSelectedProfiles(prev => {
- const next = new Set(prev);
- if (next.has(profileKey)) {
- next.delete(profileKey);
- } else {
- next.add(profileKey);
- }
- return next;
- });
- }, []);
- // Select all visible profiles
- const selectAllProfiles = useCallback(() => {
- setSelectedProfiles(new Set(filteredProfiles.map(p => getProfileKey(p))));
- }, [filteredProfiles, getProfileKey]);
- // Delete selected profiles
- const handleBulkDelete = useCallback(() => {
- if (selectedProfiles.size === 0) return;
- setShowBulkDeleteConfirm(true);
- }, [selectedProfiles.size]);
- // Execute the actual bulk delete
- const executeBulkDelete = useCallback(async () => {
- const profilesToDelete = filteredProfiles.filter(p => selectedProfiles.has(getProfileKey(p)));
- setBulkDeleteInProgress(true);
- let deleted = 0;
- for (const profile of profilesToDelete) {
- try {
- await api.deleteKProfile(selectedPrinter!, {
- slot_id: profile.slot_id,
- extruder_id: profile.extruder_id,
- nozzle_id: profile.nozzle_id,
- nozzle_diameter: profile.nozzle_diameter,
- filament_id: profile.filament_id,
- setting_id: profile.setting_id,
- });
- deleted++;
- // Small delay between deletes
- await new Promise(resolve => setTimeout(resolve, 300));
- } catch (err) {
- console.error('Failed to delete profile:', profile.name, err);
- }
- }
- showToast(t('kProfiles.toast.profilesDeleted', { count: deleted }));
- setBulkDeleteInProgress(false);
- setShowBulkDeleteConfirm(false);
- setSelectionMode(false);
- setSelectedProfiles(new Set());
- refetchProfiles();
- }, [selectedPrinter, selectedProfiles, filteredProfiles, showToast, refetchProfiles, getProfileKey, t]);
- // Generate possible keys for a profile (for notes lookup)
- // Returns array of keys to check: setting_id, slot-based, name-based
- const getProfileKeys = useCallback((profile: KProfile): string[] => {
- const keys: string[] = [];
- if (profile.setting_id) {
- keys.push(profile.setting_id);
- }
- // Slot-based key (for profiles without setting_id)
- keys.push(`slot_${profile.slot_id}_${profile.filament_id}_${profile.extruder_id}`);
- // Name-based key (for newly created profiles)
- keys.push(`name_${profile.name}_${profile.filament_id}`);
- return keys;
- }, []);
- // Save note for a profile
- const handleSaveNote = useCallback(async (profileKey: string, noteText: string) => {
- if (!selectedPrinter) return;
- try {
- await api.setKProfileNote(selectedPrinter, profileKey, noteText);
- refetchNotes();
- } catch (err) {
- console.error('Failed to save note:', err);
- showToast(t('kProfiles.toast.failedToSaveNote'), 'error');
- }
- }, [selectedPrinter, refetchNotes, showToast, t]);
- // Get note for a profile (checks all possible keys)
- // Returns { note, key } so we know which key the note was stored under
- const getNoteWithKey = useCallback((profile: KProfile): { note: string; key: string | null } => {
- if (!notesData?.notes) return { note: '', key: null };
- const keys = getProfileKeys(profile);
- for (const key of keys) {
- if (notesData.notes[key]) {
- return { note: notesData.notes[key], key };
- }
- }
- return { note: '', key: null };
- }, [notesData, getProfileKeys]);
- // Simple getter for display purposes
- const getNote = useCallback((profile: KProfile) => {
- return getNoteWithKey(profile).note;
- }, [getNoteWithKey]);
- if (printersLoading) {
- return (
- <div className="flex justify-center py-12">
- <Loader2 className="w-8 h-8 text-bambu-green animate-spin" />
- </div>
- );
- }
- if (!printers || printers.length === 0) {
- return (
- <Card>
- <CardContent className="py-12 text-center">
- <AlertCircle className="w-12 h-12 text-bambu-gray mx-auto mb-4" />
- <h3 className="text-lg font-semibold text-white mb-2">{t('kProfiles.noPrintersConfigured')}</h3>
- <p className="text-bambu-gray">
- {t('kProfiles.addPrinterInSettings')}
- </p>
- </CardContent>
- </Card>
- );
- }
- if (connectedPrinters.length === 0) {
- return (
- <Card>
- <CardContent className="py-12 text-center">
- <Printer className="w-12 h-12 text-bambu-gray mx-auto mb-4" />
- <h3 className="text-lg font-semibold text-white mb-2">{t('kProfiles.noActivePrinters')}</h3>
- <p className="text-bambu-gray">
- {t('kProfiles.enablePrinterConnection')}
- </p>
- </CardContent>
- </Card>
- );
- }
- return (
- <>
- {/* Loading overlay when refetching profiles (not initial load) */}
- {isFetching && !kprofilesLoading && (
- <div className="fixed inset-0 bg-black/50 flex flex-col items-center justify-center z-40">
- <Loader2 className="w-10 h-10 text-bambu-green animate-spin mb-3" />
- <p className="text-white font-medium">{t('kProfiles.loadingProfiles')}</p>
- </div>
- )}
- {/* Printer & Nozzle Selector */}
- <div className="flex flex-wrap gap-4 mb-6">
- <div className="flex-1 min-w-48">
- <label className="block text-sm text-bambu-gray mb-1">{t('kProfiles.printer')}</label>
- <select
- value={selectedPrinter || ''}
- onChange={(e) => setSelectedPrinter(parseInt(e.target.value))}
- className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
- >
- {connectedPrinters.map((printer) => (
- <option key={printer.id} value={printer.id}>
- {printer.name}
- </option>
- ))}
- </select>
- </div>
- <div className="w-32">
- <label className="block text-sm text-bambu-gray mb-1">{t('kProfiles.nozzle')}</label>
- <select
- value={nozzleDiameter}
- onChange={(e) => setNozzleDiameter(e.target.value)}
- className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
- >
- <option value="0.2">0.2mm</option>
- <option value="0.4">0.4mm</option>
- <option value="0.6">0.6mm</option>
- <option value="0.8">0.8mm</option>
- </select>
- </div>
- <div className="flex items-end gap-2">
- <Button
- variant="secondary"
- onClick={() => refetchProfiles()}
- disabled={isFetching || !hasPermission('kprofiles:read')}
- title={!hasPermission('kprofiles:read') ? t('kProfiles.permission.noRead') : undefined}
- >
- <RefreshCw className={`w-4 h-4 ${isFetching ? 'animate-spin' : ''}`} />
- {t('kProfiles.refresh')}
- </Button>
- <Button
- onClick={() => setShowAddModal(true)}
- disabled={!hasPermission('kprofiles:create')}
- title={!hasPermission('kprofiles:create') ? t('kProfiles.permission.noCreate') : undefined}
- >
- <Plus className="w-4 h-4" />
- {t('kProfiles.addProfile')}
- </Button>
- </div>
- </div>
- {/* Search & Filter Row */}
- <div className="flex flex-wrap gap-4 mb-4">
- <div className="flex-1 min-w-48 relative">
- <Search className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray" />
- <input
- type="text"
- value={searchQuery}
- onChange={(e) => setSearchQuery(e.target.value)}
- placeholder={t('kProfiles.searchPlaceholder')}
- className="w-full pl-10 pr-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white placeholder-bambu-gray focus:border-bambu-green focus:outline-none"
- />
- </div>
- {isDualNozzle && (
- <div className="w-36">
- <select
- value={extruderFilter}
- onChange={(e) => setExtruderFilter(e.target.value as ExtruderFilter)}
- className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
- >
- <option value="all">{t('kProfiles.allExtruders')}</option>
- <option value="left">{t('kProfiles.leftOnly')}</option>
- <option value="right">{t('kProfiles.rightOnly')}</option>
- </select>
- </div>
- )}
- <div className="w-32">
- <select
- value={flowTypeFilter}
- onChange={(e) => setFlowTypeFilter(e.target.value as FlowTypeFilter)}
- className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
- >
- <option value="all">{t('kProfiles.allFlow')}</option>
- <option value="hf">{t('kProfiles.hfOnly')}</option>
- <option value="s">{t('kProfiles.sOnly')}</option>
- </select>
- </div>
- <div className="w-32">
- <select
- value={sortOption}
- onChange={(e) => setSortOption(e.target.value as SortOption)}
- className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
- >
- <option value="name">{t('kProfiles.sortName')}</option>
- <option value="k_value">{t('kProfiles.sortKValue')}</option>
- <option value="filament">{t('kProfiles.sortFilament')}</option>
- </select>
- </div>
- </div>
- {/* Toolbar Row */}
- <div className="flex flex-wrap gap-2 mb-6">
- <Button
- variant="secondary"
- onClick={handleExport}
- disabled={!kprofiles?.profiles?.length || !hasPermission('kprofiles:read')}
- title={!hasPermission('kprofiles:read') ? t('kProfiles.permission.noExport') : undefined}
- >
- <Download className="w-4 h-4" />
- {t('kProfiles.export')}
- </Button>
- <Button
- variant="secondary"
- onClick={handleImport}
- disabled={!hasPermission('kprofiles:create')}
- title={!hasPermission('kprofiles:create') ? t('kProfiles.permission.noImport') : undefined}
- >
- <Upload className="w-4 h-4" />
- {t('kProfiles.import')}
- </Button>
- <div className="flex-1" />
- {selectionMode ? (
- <>
- <Button
- variant="secondary"
- onClick={selectAllProfiles}
- >
- <CheckSquare className="w-4 h-4" />
- {t('kProfiles.selectAll')}
- </Button>
- <Button
- variant="secondary"
- onClick={handleBulkDelete}
- disabled={selectedProfiles.size === 0 || !hasPermission('kprofiles:delete')}
- className="text-red-500 hover:bg-red-500/10"
- title={!hasPermission('kprofiles:delete') ? t('kProfiles.permission.noDelete') : undefined}
- >
- <Trash2 className="w-4 h-4" />
- {t('kProfiles.delete')} ({selectedProfiles.size})
- </Button>
- <Button
- variant="secondary"
- onClick={() => {
- setSelectionMode(false);
- setSelectedProfiles(new Set());
- }}
- >
- <X className="w-4 h-4" />
- {t('common.cancel')}
- </Button>
- </>
- ) : (
- <Button
- variant="secondary"
- onClick={() => setSelectionMode(true)}
- disabled={!filteredProfiles.length || !hasPermission('kprofiles:delete')}
- title={!hasPermission('kprofiles:delete') ? t('kProfiles.permission.noDelete') : undefined}
- >
- <CheckSquare className="w-4 h-4" />
- {t('kProfiles.select')}
- </Button>
- )}
- </div>
- {/* K-Profiles Grid */}
- {kprofilesLoading ? (
- <div className="flex justify-center py-12">
- <Loader2 className="w-8 h-8 text-bambu-green animate-spin" />
- </div>
- ) : isOfflineError ? (
- <Card>
- <CardContent className="py-12 text-center">
- <WifiOff className="w-12 h-12 text-bambu-gray mx-auto mb-4" />
- <h3 className="text-lg font-semibold text-white mb-2">{t('kProfiles.printerOffline')}</h3>
- <p className="text-bambu-gray mb-4">
- {t('kProfiles.printerOfflineDesc')}
- </p>
- <Button variant="secondary" onClick={() => refetchProfiles()}>
- <RefreshCw className="w-4 h-4" />
- {t('common.refresh')}
- </Button>
- </CardContent>
- </Card>
- ) : filteredProfiles.length > 0 ? (
- isDualNozzle ? (
- // Dual-nozzle: show Left/Right columns
- <div className="grid grid-cols-2 gap-4">
- {/* Left Extruder (extruder_id 1 on Bambu) */}
- <div>
- <h3 className="text-sm font-medium text-bambu-gray mb-2 px-1">{t('kProfiles.leftExtruder')}</h3>
- <div className="space-y-1">
- {filteredProfiles
- .filter((p) => p.extruder_id === 1)
- .map((profile) => (
- <KProfileCard
- key={getProfileKey(profile)}
- profile={profile}
- onEdit={() => setEditingProfile(profile)}
- onCopy={() => setCopyingProfile(profile)}
- selectionMode={selectionMode}
- isSelected={selectedProfiles.has(getProfileKey(profile))}
- onToggleSelect={() => toggleProfileSelection(getProfileKey(profile))}
- note={getNote(profile)}
- />
- ))}
- </div>
- </div>
- {/* Right Extruder (extruder_id 0 on Bambu) */}
- <div>
- <h3 className="text-sm font-medium text-bambu-gray mb-2 px-1">{t('kProfiles.rightExtruder')}</h3>
- <div className="space-y-1">
- {filteredProfiles
- .filter((p) => p.extruder_id === 0)
- .map((profile) => (
- <KProfileCard
- key={getProfileKey(profile)}
- profile={profile}
- onEdit={() => setEditingProfile(profile)}
- onCopy={() => setCopyingProfile(profile)}
- selectionMode={selectionMode}
- isSelected={selectedProfiles.has(getProfileKey(profile))}
- onToggleSelect={() => toggleProfileSelection(getProfileKey(profile))}
- note={getNote(profile)}
- />
- ))}
- </div>
- </div>
- </div>
- ) : (
- // Single-nozzle: show all profiles in one list
- <div className="space-y-1">
- {filteredProfiles.map((profile) => (
- <KProfileCard
- key={getProfileKey(profile)}
- profile={profile}
- onEdit={() => setEditingProfile(profile)}
- onCopy={() => setCopyingProfile(profile)}
- selectionMode={selectionMode}
- isSelected={selectedProfiles.has(getProfileKey(profile))}
- onToggleSelect={() => toggleProfileSelection(getProfileKey(profile))}
- note={getNote(profile)}
- />
- ))}
- </div>
- )
- ) : searchQuery || extruderFilter !== 'all' || flowTypeFilter !== 'all' ? (
- <Card>
- <CardContent className="py-12 text-center">
- <Search className="w-12 h-12 text-bambu-gray mx-auto mb-4" />
- <h3 className="text-lg font-semibold text-white mb-2">{t('kProfiles.noMatchingProfiles')}</h3>
- <p className="text-bambu-gray">
- {t('kProfiles.noMatchingProfilesDesc')}
- </p>
- </CardContent>
- </Card>
- ) : (
- <Card>
- <CardContent className="py-12 text-center">
- <Gauge className="w-12 h-12 text-bambu-gray mx-auto mb-4" />
- <h3 className="text-lg font-semibold text-white mb-2">{t('kProfiles.noKProfiles')}</h3>
- <p className="text-bambu-gray mb-4">
- {t('kProfiles.noKProfilesDesc', { diameter: nozzleDiameter })}
- </p>
- <Button onClick={() => setShowAddModal(true)}>
- <Plus className="w-4 h-4" />
- {t('kProfiles.createFirstProfile')}
- </Button>
- </CardContent>
- </Card>
- )}
- {/* Edit Modal */}
- {editingProfile && selectedPrinter && (() => {
- const { note, key } = getNoteWithKey(editingProfile);
- return (
- <KProfileModal
- profile={editingProfile}
- printerId={selectedPrinter}
- nozzleDiameter={nozzleDiameter}
- existingProfiles={allProfiles?.profiles || kprofiles?.profiles}
- builtinFilaments={enrichedBuiltinFilaments}
- isDualNozzle={isDualNozzle}
- initialNote={note}
- initialNoteKey={key}
- onSaveNote={handleSaveNote}
- hasPermission={hasPermission}
- onClose={() => {
- console.log('[KProfiles] Edit modal onClose - refetching profiles...');
- setEditingProfile(null);
- refetchProfiles(); // Refetch after close (handles delete case)
- }}
- onSave={() => {
- setEditingProfile(null);
- refetchProfiles();
- }}
- />
- );
- })()}
- {/* Add Modal */}
- {showAddModal && selectedPrinter && (
- <KProfileModal
- printerId={selectedPrinter}
- nozzleDiameter={nozzleDiameter}
- existingProfiles={allProfiles?.profiles || kprofiles?.profiles}
- builtinFilaments={enrichedBuiltinFilaments}
- isDualNozzle={isDualNozzle}
- onSaveNote={handleSaveNote}
- hasPermission={hasPermission}
- onClose={() => {
- setShowAddModal(false);
- refetchProfiles(); // Refetch after close
- }}
- onSave={() => {
- setShowAddModal(false);
- refetchProfiles();
- }}
- />
- )}
- {/* Copy Modal - opens add modal with prefilled values from source profile */}
- {copyingProfile && selectedPrinter && (
- <KProfileModal
- printerId={selectedPrinter}
- nozzleDiameter={nozzleDiameter}
- existingProfiles={allProfiles?.profiles || kprofiles?.profiles}
- builtinFilaments={enrichedBuiltinFilaments}
- isDualNozzle={isDualNozzle}
- onSaveNote={handleSaveNote}
- hasPermission={hasPermission}
- // Pass profile data but without slot_id to create a new profile
- profile={{
- ...copyingProfile,
- slot_id: 0, // Force new profile creation
- name: `${copyingProfile.name} (Copy)`, // Indicate it's a copy
- }}
- onClose={() => {
- setCopyingProfile(null);
- refetchProfiles();
- }}
- onSave={() => {
- setCopyingProfile(null);
- refetchProfiles();
- }}
- />
- )}
- {/* Bulk Delete Confirmation Modal */}
- {showBulkDeleteConfirm && (
- <div className="fixed inset-0 bg-black/50 flex items-center justify-center z-50 p-4">
- <Card className="w-full max-w-sm">
- <CardContent className="p-6">
- <div className="flex items-center gap-3 mb-4">
- <div className="w-10 h-10 rounded-full bg-red-500/20 flex items-center justify-center">
- <Trash2 className="w-5 h-5 text-red-500" />
- </div>
- <div>
- <h3 className="text-lg font-semibold text-white">{t('kProfiles.bulkDelete.title')}</h3>
- <p className="text-sm text-bambu-gray">{t('kProfiles.bulkDelete.cannotUndo')}</p>
- </div>
- </div>
- <p className="text-bambu-gray mb-6">
- {t('kProfiles.bulkDelete.message', { count: selectedProfiles.size })}
- </p>
- <div className="flex gap-3">
- <Button
- variant="secondary"
- onClick={() => setShowBulkDeleteConfirm(false)}
- disabled={bulkDeleteInProgress}
- className="flex-1"
- >
- {t('common.cancel')}
- </Button>
- <Button
- onClick={executeBulkDelete}
- disabled={bulkDeleteInProgress}
- className="flex-1 bg-red-500 hover:bg-red-600 text-white"
- >
- {bulkDeleteInProgress ? (
- <Loader2 className="w-4 h-4 animate-spin" />
- ) : (
- <Trash2 className="w-4 h-4" />
- )}
- {t('common.delete')}
- </Button>
- </div>
- </CardContent>
- </Card>
- </div>
- )}
- </>
- );
- }
|