| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489 |
- import { useEffect, useMemo, useState } from 'react';
- import { useTranslation } from 'react-i18next';
- import { X, Loader2, Printer, CheckSquare, Square, Search } from 'lucide-react';
- import { api, type SpoolLabelTemplate, type InventorySpool } from '../api/client';
- import { Button } from './Button';
- import { useToast } from '../contexts/ToastContext';
- /** Subset of InventorySpool the modal needs for checkbox rendering. */
- type SpoolForLabel = Pick<
- InventorySpool,
- 'id' | 'material' | 'subtype' | 'brand' | 'color_name' | 'rgba'
- >;
- interface LabelTemplatePickerModalProps {
- isOpen: boolean;
- onClose: () => void;
- /** All spools the modal can choose from. Typically the page's current
- * filter result so the modal stays consistent with what the user sees. */
- availableSpools: SpoolForLabel[];
- /** IDs to pre-check when the modal opens. Per-card icon passes a single ID;
- * the bulk header button passes every visible ID so the user lands in
- * "all checked" and refines downward. */
- initialSelectedIds: number[];
- spoolmanMode: boolean;
- }
- interface TemplateOption {
- value: SpoolLabelTemplate;
- i18nKey: string;
- fallbackLabel: string;
- fallbackHint: string;
- }
- const TEMPLATE_OPTIONS: TemplateOption[] = [
- {
- value: 'ams_holder_74x33',
- i18nKey: 'amsHolderSmall',
- fallbackLabel: 'AMS holder — small (74 × 33 mm)',
- fallbackHint: 'Single label per page; matches the printable label from MakerWorld model 752566 (AMS Filament Label Holder).',
- },
- {
- value: 'ams_holder_75x55',
- i18nKey: 'amsHolderLarge',
- fallbackLabel: 'AMS holder — large (75 × 55 mm)',
- fallbackHint: 'Single label per page; fits the cardstock-insert variant of the AMS Filament Label Holder. Roomy enough for swatch, brand, material, ID, and QR code.',
- },
- {
- value: 'box_40x30',
- i18nKey: 'box40x30',
- fallbackLabel: 'Box label (40 × 30 mm)',
- fallbackHint: 'Single label per page; common DK/Brother roll size, good for filament-bag and storage-bin labels.',
- },
- {
- value: 'box_62x29',
- i18nKey: 'box',
- fallbackLabel: 'Box label (62 × 29 mm)',
- fallbackHint: 'Single label per page; sized for Brother PT/QL and Dymo small labels.',
- },
- {
- value: 'avery_l7160',
- i18nKey: 'averyL7160',
- fallbackLabel: 'Avery L7160 — A4 sheet (38.1 × 63.5 mm × 21)',
- fallbackHint: 'EU sheet stock; 21 labels per A4 page.',
- },
- {
- value: 'avery_5160',
- i18nKey: 'avery5160',
- fallbackLabel: 'Avery 5160 — US Letter sheet (25.4 × 66.7 mm × 30)',
- fallbackHint: 'US sheet stock; 30 labels per Letter page.',
- },
- ];
- function openBlobInNewTab(blob: Blob): void {
- const url = window.URL.createObjectURL(blob);
- const win = window.open(url, '_blank', 'noopener,noreferrer');
- if (!win) {
- const a = document.createElement('a');
- a.href = url;
- a.download = 'bambuddy-labels.pdf';
- document.body.appendChild(a);
- a.click();
- document.body.removeChild(a);
- }
- setTimeout(() => window.URL.revokeObjectURL(url), 60_000);
- }
- function swatchStyle(rgba: string | null | undefined): React.CSSProperties {
- if (!rgba) return { backgroundColor: '#808080' };
- const cleaned = rgba.replace(/^#/, '').slice(0, 6);
- return cleaned.length === 6 ? { backgroundColor: `#${cleaned}` } : { backgroundColor: '#808080' };
- }
- function spoolDisplayName(s: SpoolForLabel): string {
- const head = s.color_name ?? `${s.material}${s.subtype ? ` ${s.subtype}` : ''}`;
- const brand = s.brand ? ` · ${s.brand}` : '';
- return `${head}${brand}`;
- }
- /** Build a lowercased haystack that the search input matches against. */
- function searchableText(s: SpoolForLabel): string {
- return [s.color_name, s.material, s.subtype, s.brand, `#${s.id}`]
- .filter(Boolean)
- .join(' ')
- .toLowerCase();
- }
- type SortMode = 'id' | 'color';
- /** Sort key for the "by colour" mode (#1410).
- *
- * Returns a 2-tuple so JS array compare does the right thing without us having
- * to spell out a comparator: ``[bucket, position]``. Chromatic colours
- * (saturation above the threshold) go in bucket 0 ordered by HSL hue, so the
- * sheet reads as a continuous rainbow. Achromatic colours (white / grey /
- * black, plus missing/invalid rgba) go in bucket 1 ordered by lightness so the
- * neutrals trail at the end of the rainbow going dark → light. Multi-colour
- * spools sort on their primary ``rgba``; their ``extra_colors`` stripe is
- * still rendered on the label itself but doesn't drive the sort.
- */
- function colorSortKey(rgba: string | null | undefined): [number, number] {
- if (!rgba) return [1, 0]; // unknown colour — bucket with the neutrals at black
- const cleaned = rgba.replace(/^#/, '').slice(0, 6);
- if (cleaned.length !== 6) return [1, 0];
- const r = parseInt(cleaned.slice(0, 2), 16);
- const g = parseInt(cleaned.slice(2, 4), 16);
- const b = parseInt(cleaned.slice(4, 6), 16);
- if ([r, g, b].some(Number.isNaN)) return [1, 0];
- const rn = r / 255;
- const gn = g / 255;
- const bn = b / 255;
- const max = Math.max(rn, gn, bn);
- const min = Math.min(rn, gn, bn);
- const l = (max + min) / 2;
- const delta = max - min;
- // Saturation in the HSL definition. Achromatic cutoff at 0.1 is generous —
- // matches what feels "grey enough" to a user picking colours, without
- // sending dark muted colours like deep navy into the neutrals bucket.
- const s = delta === 0 ? 0 : delta / (1 - Math.abs(2 * l - 1));
- if (s < 0.1) return [1, l]; // neutrals: ordered black → white
- let h = 0;
- if (max === rn) h = ((gn - bn) / delta) % 6;
- else if (max === gn) h = (bn - rn) / delta + 2;
- else h = (rn - gn) / delta + 4;
- h = h * 60;
- if (h < 0) h += 360;
- return [0, h]; // chromatic: ordered by hue 0..360
- }
- export function LabelTemplatePickerModal({
- isOpen,
- onClose,
- availableSpools,
- initialSelectedIds,
- spoolmanMode,
- }: LabelTemplatePickerModalProps) {
- const { t } = useTranslation();
- const { showToast } = useToast();
- const [pending, setPending] = useState<SpoolLabelTemplate | null>(null);
- const [selectedIds, setSelectedIds] = useState<Set<number>>(new Set());
- const [search, setSearch] = useState('');
- const [materialFilter, setMaterialFilter] = useState<string>('');
- const [sortMode, setSortMode] = useState<SortMode>('id');
- // Sync from caller and reset transient state on open. Intentionally not
- // reactive to props while open — once the user starts editing we don't want
- // a parent re-render to clobber their selection / filter / search.
- useEffect(() => {
- if (isOpen) {
- const allowed = new Set(availableSpools.map((s) => s.id));
- setSelectedIds(new Set(initialSelectedIds.filter((id) => allowed.has(id))));
- setSearch('');
- setMaterialFilter('');
- setSortMode('id');
- setPending(null);
- }
- // eslint-disable-next-line react-hooks/exhaustive-deps
- }, [isOpen]);
- const sortedSpools = useMemo(() => {
- const copy = [...availableSpools];
- if (sortMode === 'color') {
- copy.sort((a, b) => {
- const ka = colorSortKey(a.rgba);
- const kb = colorSortKey(b.rgba);
- if (ka[0] !== kb[0]) return ka[0] - kb[0];
- if (ka[1] !== kb[1]) return ka[1] - kb[1];
- // Stable tiebreaker on ID so identical colours print in a deterministic
- // order across renders.
- return a.id - b.id;
- });
- return copy;
- }
- copy.sort((a, b) => a.id - b.id);
- return copy;
- }, [availableSpools, sortMode]);
- // Material chips are derived from the *full* available set so they stay
- // stable when search/material filter narrows the visible list.
- const materials = useMemo(() => {
- const set = new Set<string>();
- for (const s of sortedSpools) {
- if (s.material) set.add(s.material.toUpperCase());
- }
- return [...set].sort();
- }, [sortedSpools]);
- const visibleSpools = useMemo(() => {
- const q = search.trim().toLowerCase();
- return sortedSpools.filter((s) => {
- if (materialFilter && (s.material || '').toUpperCase() !== materialFilter) return false;
- if (q && !searchableText(s).includes(q)) return false;
- return true;
- });
- }, [sortedSpools, search, materialFilter]);
- const allVisibleChecked =
- visibleSpools.length > 0 && visibleSpools.every((s) => selectedIds.has(s.id));
- if (!isOpen) return null;
- const selectedCount = selectedIds.size;
- const noSelection = selectedCount === 0;
- function toggleOne(id: number) {
- setSelectedIds((prev) => {
- const next = new Set(prev);
- if (next.has(id)) next.delete(id);
- else next.add(id);
- return next;
- });
- }
- function selectAllVisible() {
- setSelectedIds((prev) => {
- const next = new Set(prev);
- for (const s of visibleSpools) next.add(s.id);
- return next;
- });
- }
- function deselectVisible() {
- setSelectedIds((prev) => {
- const next = new Set(prev);
- for (const s of visibleSpools) next.delete(s.id);
- return next;
- });
- }
- function clearAll() {
- setSelectedIds(new Set());
- }
- async function handlePick(template: SpoolLabelTemplate) {
- if (noSelection || pending) return;
- // Order matters: the backend (labels.py) prints labels in the same order
- // we send IDs. Use the sorted list so a "by colour" sort flows through to
- // the PDF instead of being clobbered by an ascending-ID re-sort.
- const ids = sortedSpools.filter((s) => selectedIds.has(s.id)).map((s) => s.id);
- setPending(template);
- try {
- const blob = spoolmanMode
- ? await api.printSpoolmanSpoolLabels({ spool_ids: ids, template })
- : await api.printSpoolLabels({ spool_ids: ids, template });
- openBlobInNewTab(blob);
- onClose();
- } catch (err) {
- const msg = err instanceof Error ? err.message : String(err);
- showToast(
- t('inventory.labels.error', 'Could not generate labels: {{msg}}', { msg }),
- 'error',
- );
- } finally {
- setPending(null);
- }
- }
- return (
- <div className="fixed inset-0 z-50 flex items-start sm:items-center justify-center p-4 overflow-y-auto">
- <div
- className="absolute inset-0 bg-black/60 backdrop-blur-sm"
- onClick={onClose}
- />
- <div className="relative w-full max-w-3xl bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-xl shadow-2xl max-h-[90vh] overflow-hidden flex flex-col my-auto">
- {/* Header */}
- <div className="flex items-center justify-between p-4 border-b border-bambu-dark-tertiary">
- <div className="flex items-center gap-2">
- <Printer className="w-5 h-5 text-bambu-green" />
- <h2 className="text-lg font-semibold text-white">
- {t('inventory.labels.title', 'Print spool labels')}
- </h2>
- {selectedCount > 0 && (
- <span className="text-sm text-bambu-gray">
- ({t('inventory.labels.selectedCount', '{{count}} selected', { count: selectedCount })})
- </span>
- )}
- </div>
- <button
- onClick={onClose}
- className="p-1 text-bambu-gray hover:text-white rounded transition-colors"
- aria-label={t('common.close', 'Close')}
- >
- <X className="w-5 h-5" />
- </button>
- </div>
- {/* Search + material chips */}
- <div className="p-4 space-y-2 border-b border-bambu-dark-tertiary">
- <div className="relative">
- <Search className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray pointer-events-none" />
- <input
- type="search"
- value={search}
- onChange={(e) => setSearch(e.target.value)}
- placeholder={t('inventory.labels.searchPlaceholder', 'Search name, brand, or #ID')}
- className="w-full pl-9 pr-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white text-sm placeholder:text-bambu-gray focus:outline-none focus:border-bambu-green"
- />
- </div>
- {materials.length > 1 && (
- <div className="flex flex-wrap items-center gap-1.5">
- <span className="text-xs text-bambu-gray mr-1">
- {t('inventory.labels.filterByMaterial', 'Material:')}
- </span>
- <button
- type="button"
- onClick={() => setMaterialFilter('')}
- className={`px-2 py-0.5 text-xs rounded-full border transition ${
- materialFilter === ''
- ? 'bg-bambu-green text-bambu-dark border-bambu-green'
- : 'bg-bambu-dark text-bambu-gray border-bambu-dark-tertiary hover:border-bambu-gray'
- }`}
- >
- {t('inventory.labels.allMaterials', 'All')}
- </button>
- {materials.map((m) => (
- <button
- key={m}
- type="button"
- onClick={() => setMaterialFilter(m)}
- className={`px-2 py-0.5 text-xs rounded-full border transition ${
- materialFilter === m
- ? 'bg-bambu-green text-bambu-dark border-bambu-green'
- : 'bg-bambu-dark text-bambu-gray border-bambu-dark-tertiary hover:border-bambu-gray'
- }`}
- >
- {m}
- </button>
- ))}
- </div>
- )}
- <div className="flex flex-wrap items-center gap-1.5">
- <span className="text-xs text-bambu-gray mr-1">
- {t('inventory.labels.sortBy.label')}
- </span>
- <button
- type="button"
- onClick={() => setSortMode('id')}
- className={`px-2 py-0.5 text-xs rounded-full border transition ${
- sortMode === 'id'
- ? 'bg-bambu-green text-bambu-dark border-bambu-green'
- : 'bg-bambu-dark text-bambu-gray border-bambu-dark-tertiary hover:border-bambu-gray'
- }`}
- >
- {t('inventory.labels.sortBy.id')}
- </button>
- <button
- type="button"
- onClick={() => setSortMode('color')}
- className={`px-2 py-0.5 text-xs rounded-full border transition ${
- sortMode === 'color'
- ? 'bg-bambu-green text-bambu-dark border-bambu-green'
- : 'bg-bambu-dark text-bambu-gray border-bambu-dark-tertiary hover:border-bambu-gray'
- }`}
- >
- {t('inventory.labels.sortBy.color')}
- </button>
- </div>
- </div>
- {/* Action bar */}
- <div className="px-4 pt-3 pb-2 flex items-center justify-between gap-3 flex-wrap">
- <span className="text-sm text-bambu-gray">
- {t('inventory.labels.pickSpools', 'Pick which spools to print labels for:')}
- </span>
- <div className="flex items-center gap-3 text-xs">
- <button
- type="button"
- onClick={allVisibleChecked ? deselectVisible : selectAllVisible}
- disabled={visibleSpools.length === 0}
- className="text-bambu-green hover:underline disabled:opacity-50 disabled:no-underline disabled:cursor-not-allowed"
- >
- {allVisibleChecked
- ? t('inventory.labels.deselectVisible', 'Deselect visible')
- : t('inventory.labels.selectVisible', 'Select all visible ({{count}})', {
- count: visibleSpools.length,
- })}
- </button>
- <button
- type="button"
- onClick={clearAll}
- disabled={selectedCount === 0}
- className="text-bambu-gray hover:text-white hover:underline disabled:opacity-50 disabled:no-underline disabled:cursor-not-allowed"
- >
- {t('inventory.labels.clearAll', 'Clear all')}
- </button>
- </div>
- </div>
- {/* Spool list */}
- <div className="flex-1 overflow-y-auto px-2 pb-2 min-h-0">
- {visibleSpools.length === 0 ? (
- <div className="text-center text-sm text-bambu-gray py-6">
- {sortedSpools.length === 0
- ? t('inventory.labels.noSpoolsToShow', 'No spools to show. Adjust your filter and try again.')
- : t('inventory.labels.noMatches', 'No spools match the current search or filter.')}
- </div>
- ) : (
- <ul className="space-y-0.5">
- {visibleSpools.map((s) => {
- const checked = selectedIds.has(s.id);
- return (
- <li key={s.id}>
- <label className="flex items-center gap-3 px-2 py-1.5 rounded hover:bg-bambu-dark-tertiary/50 cursor-pointer">
- {checked ? (
- <CheckSquare className="w-4 h-4 text-bambu-green shrink-0" />
- ) : (
- <Square className="w-4 h-4 text-bambu-gray shrink-0" />
- )}
- <input
- type="checkbox"
- checked={checked}
- onChange={() => toggleOne(s.id)}
- className="sr-only"
- />
- <span
- className="w-4 h-4 rounded border border-black/20 shrink-0"
- style={swatchStyle(s.rgba)}
- />
- <span className="flex-1 min-w-0 truncate text-sm text-white">
- {spoolDisplayName(s)}
- </span>
- <span className="text-xs font-mono text-bambu-gray shrink-0">
- #{s.id}
- </span>
- </label>
- </li>
- );
- })}
- </ul>
- )}
- </div>
- {/* Templates — 2x2 grid on >= sm so all 4 plus the Cancel footer fit
- inside max-h-[90vh] even when browser chrome eats into the viewport
- (#1230). Stacked single column on mobile widths. */}
- <div className="px-3 pt-2 pb-2 grid grid-cols-1 sm:grid-cols-2 gap-2 border-t border-bambu-dark-tertiary">
- {TEMPLATE_OPTIONS.map((opt) => {
- const isPending = pending === opt.value;
- const label = t(`inventory.labels.templates.${opt.i18nKey}.label`, opt.fallbackLabel);
- const hint = t(`inventory.labels.templates.${opt.i18nKey}.hint`, opt.fallbackHint);
- return (
- <button
- key={opt.value}
- disabled={noSelection || pending !== null}
- onClick={() => handlePick(opt.value)}
- title={`${label} — ${hint}`}
- className="w-full text-left p-2.5 rounded-lg border border-bambu-dark-tertiary bg-bambu-dark hover:border-bambu-green hover:bg-bambu-green/10 disabled:opacity-50 disabled:cursor-not-allowed disabled:hover:border-bambu-dark-tertiary disabled:hover:bg-bambu-dark transition flex items-center gap-3"
- >
- <div className="flex-1 min-w-0">
- <div className="font-medium text-white text-sm truncate">{label}</div>
- <div className="text-xs text-bambu-gray mt-0.5 truncate">{hint}</div>
- </div>
- {isPending && <Loader2 className="w-4 h-4 animate-spin text-bambu-green shrink-0" />}
- </button>
- );
- })}
- </div>
- <div className="flex justify-end gap-2 px-5 py-2 border-t border-bambu-dark-tertiary">
- <Button variant="secondary" onClick={onClose} disabled={pending !== null}>
- {t('common.cancel', 'Cancel')}
- </Button>
- </div>
- </div>
- </div>
- );
- }
|