| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620 |
- import { useState, useRef, useEffect, useLayoutEffect, type ReactNode } from 'react';
- import { createPortal } from 'react-dom';
- import { useNavigate } from 'react-router-dom';
- import { useTranslation } from 'react-i18next';
- import { Droplets, Copy, Check, Settings2, Package, Unlink } from 'lucide-react';
- import { isLightColor } from '../utils/colors';
- interface FilamentData {
- vendor: 'Bambu Lab' | 'Generic';
- profile: string;
- colorName: string;
- colorHex: string | null;
- kFactor: string;
- fillLevel: number | null; // null = unknown
- trayUuid?: string | null; // Bambu Lab spool UUID for Spoolman linking
- tagUid?: string | null; // Generic NFC tag UID fallback for linking
- fillSource?: 'ams' | 'spoolman' | 'inventory'; // Source of fill level data
- }
- interface SpoolmanConfig {
- enabled: boolean;
- onLinkSpool?: () => void;
- onUnlinkSpool?: () => void;
- linkedSpoolId?: number | null; // Spoolman spool ID if this tray is already linked
- spoolmanUrl?: string | null; // Base URL for Spoolman (for "Open in Spoolman" link)
- syncMode?: string | null; // If auto-sync is enabled, we may want to hide the unlink option for Bambu spools
- }
- interface InventoryConfig {
- onAssignSpool?: () => void;
- onUnassignSpool?: () => void;
- assignedSpool?: { id: number; material: string; brand: string | null; color_name: string | null; remainingWeightGrams?: number | null } | null;
- isAssigned?: boolean;
- }
- interface ConfigureSlotConfig {
- enabled: boolean;
- onConfigure?: () => void;
- }
- interface FilamentHoverCardProps {
- data: FilamentData;
- children: ReactNode;
- disabled?: boolean;
- className?: string;
- spoolman?: SpoolmanConfig;
- inventory?: InventoryConfig;
- configureSlot?: ConfigureSlotConfig;
- }
- /**
- * A hover card that displays filament details when hovering over AMS slots.
- * Replaces the basic browser tooltip with a styled popover.
- */
- export function FilamentHoverCard({ data, children, disabled, className = '', spoolman, inventory, configureSlot }: FilamentHoverCardProps) {
- const { t } = useTranslation();
- const navigate = useNavigate();
- const [isVisible, setIsVisible] = useState(false);
- const [position, setPosition] = useState<'top' | 'bottom'>('top');
- // Screen-space coordinates for the portaled card (#1336 follow-up). Using
- // a portal + position:fixed lets the popover escape sibling printer cards
- // that create their own stacking contexts on the dashboard — without this,
- // a card later in DOM order draws over the hover popover regardless of
- // z-index because z-index doesn't cross stacking-context boundaries.
- const [coords, setCoords] = useState<{ top: number; left: number } | null>(null);
- const [copied, setCopied] = useState(false);
- const [showUnlinkConfirm, setShowUnlinkConfirm] = useState(false);
- const triggerRef = useRef<HTMLDivElement>(null);
- const cardRef = useRef<HTMLDivElement>(null);
- const timeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null);
- const handleCopyUuid = () => {
- const uuid = data.trayUuid;
- if (!uuid) return;
- // Try modern clipboard API first, fallback to execCommand
- if (navigator.clipboard && window.isSecureContext) {
- navigator.clipboard.writeText(uuid).then(() => {
- setCopied(true);
- setTimeout(() => setCopied(false), 2000);
- }).catch(() => {
- // Fallback on error
- fallbackCopy(uuid);
- });
- } else {
- fallbackCopy(uuid);
- }
- };
- const fallbackCopy = (text: string) => {
- const textarea = document.createElement('textarea');
- textarea.value = text;
- textarea.style.position = 'fixed';
- textarea.style.opacity = '0';
- document.body.appendChild(textarea);
- textarea.select();
- try {
- document.execCommand('copy');
- setCopied(true);
- setTimeout(() => setCopied(false), 2000);
- } catch {
- console.error('Failed to copy to clipboard');
- }
- document.body.removeChild(textarea);
- };
- // Compute placement (top/bottom) + screen coordinates for the portaled
- // card. Runs on visibility change, scroll, and resize so the popover
- // tracks the trigger when the viewport moves. useLayoutEffect rather
- // than useEffect so the first paint already has the correct coords —
- // avoids a one-frame flicker at (0, 0).
- useLayoutEffect(() => {
- if (!isVisible) {
- setCoords(null);
- return;
- }
- const compute = () => {
- if (!triggerRef.current || !cardRef.current) return;
- const triggerRect = triggerRef.current.getBoundingClientRect();
- const cardHeight = cardRef.current.offsetHeight;
- const cardWidth = cardRef.current.offsetWidth;
- const headerHeight = 56;
- const spaceAbove = triggerRect.top - headerHeight;
- const spaceBelow = window.innerHeight - triggerRect.bottom;
- const placement: 'top' | 'bottom' =
- spaceAbove < cardHeight + 12 && spaceBelow > spaceAbove ? 'bottom' : 'top';
- const centerX = triggerRect.left + triggerRect.width / 2;
- const left = Math.max(8, Math.min(centerX - cardWidth / 2, window.innerWidth - cardWidth - 8));
- const top = placement === 'top' ? triggerRect.top - cardHeight - 8 : triggerRect.bottom + 8;
- setPosition(placement);
- setCoords({ top, left });
- };
- // First compute is synchronous from the layout effect; a follow-up rAF
- // re-measures after the card actually has its rendered dimensions.
- compute();
- const rafId = requestAnimationFrame(compute);
- window.addEventListener('scroll', compute, true);
- window.addEventListener('resize', compute);
- return () => {
- cancelAnimationFrame(rafId);
- window.removeEventListener('scroll', compute, true);
- window.removeEventListener('resize', compute);
- };
- }, [isVisible]);
- const handleMouseEnter = () => {
- if (disabled) return;
- if (timeoutRef.current) clearTimeout(timeoutRef.current);
- // Small delay to prevent flicker on quick mouse movements
- timeoutRef.current = setTimeout(() => setIsVisible(true), 80);
- };
- const handleMouseLeave = () => {
- if (timeoutRef.current) clearTimeout(timeoutRef.current);
- timeoutRef.current = setTimeout(() => setIsVisible(false), 100);
- };
- // Cleanup timeout on unmount
- useEffect(() => {
- return () => {
- if (timeoutRef.current) clearTimeout(timeoutRef.current);
- };
- }, []);
- // Get fill bar color based on percentage
- const getFillColor = (fill: number): string => {
- if (fill <= 15) return '#ef4444'; // red
- if (fill <= 30) return '#f97316'; // orange
- if (fill <= 50) return '#eab308'; // yellow
- return '#22c55e'; // green
- };
- const colorHex = data.colorHex ? `#${data.colorHex.replace('#', '')}` : null;
- const assignedRemainingWeight = inventory?.assignedSpool?.remainingWeightGrams ?? null;
- return (
- <div
- ref={triggerRef}
- className={`relative ${className}`}
- onMouseEnter={handleMouseEnter}
- onMouseLeave={handleMouseLeave}
- >
- {children}
- {/* Portaled hover card — rendered into document.body so it escapes
- any ancestor stacking context. Sibling printer cards on the
- dashboard create their own stacking contexts; without the portal
- the popover gets covered by the next card even at z-[60]
- (#1336 follow-up). */}
- {isVisible && createPortal(
- <div
- ref={cardRef}
- className="fixed z-[60] animate-in fade-in-0 zoom-in-95 duration-150"
- style={{
- top: coords?.top ?? -9999,
- left: coords?.left ?? -9999,
- maxWidth: 'calc(100vw - 24px)',
- // Hide until coords are computed to avoid a (-9999,-9999) flash.
- visibility: coords ? 'visible' : 'hidden',
- }}
- onMouseEnter={handleMouseEnter}
- onMouseLeave={handleMouseLeave}
- >
- {/* Card container */}
- <div className="
- w-52 bg-bambu-dark-secondary border border-bambu-dark-tertiary
- rounded-lg shadow-xl overflow-hidden
- backdrop-blur-sm
- ">
- {/* Color swatch header - the hero element */}
- <div
- className="h-12 relative overflow-hidden"
- style={{
- backgroundColor: colorHex || '#3d3d3d',
- }}
- >
- {/* Subtle gradient overlay for depth */}
- <div className="absolute inset-0 bg-gradient-to-b from-white/10 to-transparent" />
- {/* Color name on swatch */}
- <div className={`
- absolute inset-0 flex items-center justify-center
- font-semibold text-sm tracking-wide
- ${isLightColor(colorHex) ? 'text-black/80' : 'text-white/90'}
- `}>
- {data.colorName}
- </div>
- {/* Vendor badge - solid background for visibility on any color */}
- <div className={`
- absolute top-1.5 right-1.5 px-1.5 py-0.5 rounded text-[9px] font-bold uppercase tracking-wider
- ${data.vendor === 'Bambu Lab'
- ? 'bg-black/60 text-white'
- : 'bg-black/50 text-white/90'}
- `}>
- {data.vendor === 'Bambu Lab' ? 'BBL' : 'GEN'}
- </div>
- </div>
- {/* Details section */}
- <div className="p-3 space-y-2.5">
- {/* Profile name */}
- <div className="flex items-center justify-between">
- <span className="text-[10px] uppercase tracking-wider text-bambu-gray font-medium">
- {t('ams.profile')}
- </span>
- <span className="text-xs text-white font-semibold truncate max-w-[120px]">
- {data.profile}
- </span>
- </div>
- {/* K Factor */}
- <div className="flex items-center justify-between">
- <span className="text-[10px] uppercase tracking-wider text-bambu-gray font-medium">
- {t('ams.kFactor')}
- </span>
- <span className="text-xs text-bambu-green font-mono font-bold">
- {data.kFactor}
- </span>
- </div>
- {/* Fill Level */}
- <div className="space-y-1">
- <div className="flex items-center justify-between">
- <span className="text-[10px] uppercase tracking-wider text-bambu-gray font-medium flex items-center gap-1">
- <Droplets className="w-3 h-3" />
- {t('ams.fill')}
- </span>
- <span className="text-xs text-white font-semibold flex items-center gap-1">
- <span>{data.fillLevel !== null ? `${data.fillLevel}%` : '—'}</span>
- {assignedRemainingWeight !== null && data.fillLevel !== null && (
- <span className="text-[9px] text-bambu-gray font-normal">• {assignedRemainingWeight}g</span>
- )}
- </span>
- </div>
- {/* Fill bar */}
- <div className="h-1.5 bg-black/40 rounded-full overflow-hidden">
- {data.fillLevel !== null ? (
- <div
- className="h-full rounded-full transition-all duration-300"
- style={{
- width: `${data.fillLevel}%`,
- backgroundColor: getFillColor(data.fillLevel),
- }}
- />
- ) : (
- <div className="h-full w-full bg-bambu-gray/30 rounded-full" />
- )}
- </div>
- </div>
- {/* Spoolman section - only show if enabled */}
- {spoolman?.enabled && (
- <div className="pt-2 mt-2 border-t border-bambu-dark-tertiary space-y-2">
- {/* Tray UUID with copy button */}
- <div className="flex items-center justify-between">
- <span className="text-[10px] uppercase tracking-wider text-bambu-gray font-medium">
- {t('spoolman.spoolId')}
- </span>
- {data.trayUuid ? (
- <button
- onClick={(e) => {
- e.stopPropagation();
- handleCopyUuid();
- }}
- className="flex items-center gap-1 text-xs text-bambu-gray hover:text-white transition-colors"
- title="Copy spool UUID"
- >
- <span className="font-mono text-[10px] truncate max-w-[80px]">
- {data.trayUuid.slice(0, 8)}...
- </span>
- {copied ? (
- <Check className="w-3 h-3 text-bambu-green" />
- ) : (
- <Copy className="w-3 h-3" />
- )}
- </button>
- ) : (
- <span className="text-[10px] text-bambu-gray">—</span>
- )}
- </div>
- {/* Open in inventory button (when already linked to a Spoolman spool) */}
- {spoolman.linkedSpoolId && (
- <>
- <button
- onClick={(e) => {
- e.stopPropagation();
- navigate(`/inventory?spool=${spoolman.linkedSpoolId}`);
- }}
- className="w-full flex items-center justify-center gap-1.5 px-2 py-1.5 text-xs font-medium rounded transition-colors bg-bambu-green/20 hover:bg-bambu-green/30 text-bambu-green"
- title={t('inventory.openInInventory')}
- >
- <Package className="w-3.5 h-3.5" />
- {t('inventory.openInInventory')}
- </button>
- </>
- )}
- {/* Link/Unlink action buttons intentionally NOT rendered
- here. The inventory section below already provides
- Assign/Unassign for slot-binding (the primary user
- flow in Spoolman mode). Showing the spoolman tag-link
- buttons in addition surfaced two red Unlink-icon
- buttons for what users perceive as the same action,
- regardless of whether the labels said "Unlink Spool"
- vs "Unassign Spool". Tag-linking remains available
- via dedicated UI (LinkSpoolModal can be opened from
- Spoolman settings / inventory page). */}
- </div>
- )}
- {/* Inventory section — shown for every vendor including
- Bambu Lab (#1133). The earlier "non-Bambu only" gate
- prevented users from manually assigning a Bambu spool
- in inventory to an AMS slot when they didn't want to
- re-scan via SpoolBuddy NFC. */}
- {inventory && (
- <div className="pt-2 mt-2 border-t border-bambu-dark-tertiary space-y-2">
- {inventory.assignedSpool ? (
- <>
- <div className="flex items-center gap-1.5">
- <Package className="w-3 h-3 text-bambu-green" />
- <span className="text-[10px] uppercase tracking-wider text-bambu-gray font-medium">
- {t('inventory.assigned')}
- </span>
- </div>
- <p className="text-xs text-white truncate">
- {inventory.assignedSpool.brand ? `${inventory.assignedSpool.brand} ` : ''}
- {inventory.assignedSpool.material}
- {inventory.assignedSpool.color_name ? ` - ${inventory.assignedSpool.color_name}` : ''}
- </p>
- {(!spoolman?.linkedSpoolId || inventory.assignedSpool!.id !== spoolman.linkedSpoolId) && (
- <button
- onClick={(e) => {
- e.stopPropagation();
- navigate(`/inventory?spool=${inventory.assignedSpool!.id}`);
- }}
- className="w-full flex items-center justify-center gap-1.5 px-2 py-1.5 text-xs font-medium rounded transition-colors bg-bambu-green/20 hover:bg-bambu-green/30 text-bambu-green"
- title={t('inventory.openInInventory')}
- >
- <Package className="w-3.5 h-3.5" />
- {t('inventory.openInInventory')}
- </button>
- )}
- {inventory.onUnassignSpool && (
- <button
- onClick={(e) => {
- e.stopPropagation();
- inventory.onUnassignSpool?.();
- }}
- className="w-full flex items-center justify-center gap-1.5 px-2 py-1.5 text-xs font-medium rounded transition-colors bg-red-500/20 hover:bg-red-500/30 text-red-400"
- >
- <Unlink className="w-3.5 h-3.5" />
- {t('inventory.unassignSpool')}
- </button>
- )}
- </>
- ) : inventory.onAssignSpool ? (
- <button
- onClick={inventory.isAssigned ? undefined : (e) => {
- e.stopPropagation();
- inventory.onAssignSpool?.();
- }}
- disabled={!!inventory.isAssigned}
- className={`w-full flex items-center justify-center gap-1.5 px-2 py-1.5 text-xs font-medium rounded transition-colors bg-bambu-blue/20 text-bambu-blue ${
- inventory.isAssigned ? 'opacity-50 cursor-not-allowed' : 'hover:bg-bambu-blue/30'
- }`}
- >
- <Package className="w-3.5 h-3.5" />
- {t('inventory.assignSpool')}
- </button>
- ) : null}
- </div>
- )}
- {/* Configure slot section - always show if enabled */}
- {configureSlot?.enabled && (
- <div className={`${spoolman?.enabled && data.trayUuid ? '' : 'pt-2 mt-2 border-t border-bambu-dark-tertiary'}`}>
- <button
- onClick={(e) => {
- e.stopPropagation();
- configureSlot.onConfigure?.();
- }}
- className="w-full flex items-center justify-center gap-1.5 px-2 py-1.5 text-xs font-medium rounded transition-colors bg-bambu-blue/20 hover:bg-bambu-blue/30 text-bambu-blue"
- title={t('ams.configureSlot')}
- >
- <Settings2 className="w-3.5 h-3.5" />
- {t('ams.configure')}
- </button>
- </div>
- )}
- </div>
- </div>
- {/* Arrow pointer */}
- <div
- className={`
- absolute left-1/2 -translate-x-1/2 w-0 h-0
- border-l-[6px] border-l-transparent
- border-r-[6px] border-r-transparent
- ${position === 'top'
- ? 'top-full border-t-[6px] border-t-bambu-dark-tertiary'
- : 'bottom-full border-b-[6px] border-b-bambu-dark-tertiary'}
- `}
- />
- </div>,
- document.body,
- )}
- {/* Unlink Confirmation Dialog */}
- {showUnlinkConfirm && (
- <div className="fixed inset-0 z-[100] flex items-center justify-center" onClick={() => setShowUnlinkConfirm(false)}>
- <div className="absolute inset-0 bg-black/60 backdrop-blur-sm" />
- <div
- className="relative bg-bambu-dark-secondary rounded-lg shadow-xl w-full max-w-sm mx-4 border border-bambu-dark-tertiary"
- onClick={(e) => e.stopPropagation()}
- >
- <div className="p-4 space-y-4">
- <div className="space-y-2">
- <h3 className="text-base font-semibold text-white">
- {t('spoolman.unlinkConfirmTitle')}
- </h3>
- <p className="text-sm text-bambu-gray">
- {t('spoolman.unlinkConfirmMessage')}
- </p>
- </div>
- <div className="flex gap-2">
- <button
- onClick={() => setShowUnlinkConfirm(false)}
- className="flex-1 px-3 py-2 text-sm font-medium rounded transition-colors bg-bambu-dark hover:bg-bambu-dark-tertiary text-white"
- >
- {t('common.cancel')}
- </button>
- <button
- onClick={() => {
- spoolman?.onUnlinkSpool?.();
- setShowUnlinkConfirm(false);
- }}
- className="flex-1 px-3 py-2 text-sm font-medium rounded transition-colors bg-red-500/20 hover:bg-red-500/30 text-red-400"
- >
- {t('inventory.unassignSpool')}
- </button>
- </div>
- </div>
- </div>
- </div>
- )}
- </div>
- );
- }
- interface EmptySlotHoverCardProps {
- children: ReactNode;
- className?: string;
- configureSlot?: ConfigureSlotConfig;
- onAssignSpool?: () => void;
- }
- export function EmptySlotHoverCard({ children, className = '', configureSlot, onAssignSpool }: EmptySlotHoverCardProps) {
- const { t } = useTranslation();
- const [isVisible, setIsVisible] = useState(false);
- // Screen-space coords for the portaled card — same pattern as
- // FilamentHoverCard, see comment there (#1336 follow-up).
- const [coords, setCoords] = useState<{ top: number; left: number } | null>(null);
- const triggerRef = useRef<HTMLDivElement>(null);
- const cardRef = useRef<HTMLDivElement>(null);
- const timeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null);
- const handleMouseEnter = () => {
- if (timeoutRef.current) clearTimeout(timeoutRef.current);
- timeoutRef.current = setTimeout(() => setIsVisible(true), 80);
- };
- const handleMouseLeave = () => {
- if (timeoutRef.current) clearTimeout(timeoutRef.current);
- timeoutRef.current = setTimeout(() => setIsVisible(false), 100);
- };
- useEffect(() => {
- return () => {
- if (timeoutRef.current) clearTimeout(timeoutRef.current);
- };
- }, []);
- useLayoutEffect(() => {
- if (!isVisible) {
- setCoords(null);
- return;
- }
- const compute = () => {
- if (!triggerRef.current || !cardRef.current) return;
- const triggerRect = triggerRef.current.getBoundingClientRect();
- const cardHeight = cardRef.current.offsetHeight;
- const cardWidth = cardRef.current.offsetWidth;
- const centerX = triggerRect.left + triggerRect.width / 2;
- const left = Math.max(8, Math.min(centerX - cardWidth / 2, window.innerWidth - cardWidth - 8));
- const top = triggerRect.top - cardHeight - 8;
- setCoords({ top, left });
- };
- compute();
- const rafId = requestAnimationFrame(compute);
- window.addEventListener('scroll', compute, true);
- window.addEventListener('resize', compute);
- return () => {
- cancelAnimationFrame(rafId);
- window.removeEventListener('scroll', compute, true);
- window.removeEventListener('resize', compute);
- };
- }, [isVisible]);
- return (
- <div
- ref={triggerRef}
- className={`relative ${className}`}
- onMouseEnter={handleMouseEnter}
- onMouseLeave={handleMouseLeave}
- >
- {children}
- {isVisible && createPortal(
- <div
- ref={cardRef}
- className="fixed z-[60] animate-in fade-in-0 zoom-in-95 duration-150"
- style={{
- top: coords?.top ?? -9999,
- left: coords?.left ?? -9999,
- visibility: coords ? 'visible' : 'hidden',
- }}
- onMouseEnter={handleMouseEnter}
- onMouseLeave={handleMouseLeave}
- >
- <div className="
- bg-bambu-dark-secondary border border-bambu-dark-tertiary
- rounded-md shadow-lg overflow-hidden
- ">
- <div className="px-3 py-1.5 text-xs text-bambu-gray whitespace-nowrap">
- {t('ams.emptySlot')}
- </div>
- {/* Configure slot button */}
- {(configureSlot?.enabled || onAssignSpool) && (
- <div className="px-2 pb-2 space-y-1">
- {configureSlot?.enabled && (
- <button
- onClick={(e) => {
- e.stopPropagation();
- configureSlot.onConfigure?.();
- }}
- className="w-full flex items-center justify-center gap-1.5 px-2 py-1.5 text-xs font-medium rounded transition-colors bg-bambu-blue/20 hover:bg-bambu-blue/30 text-bambu-blue"
- title={t('ams.configureSlot')}
- >
- <Settings2 className="w-3.5 h-3.5" />
- {t('ams.configure')}
- </button>
- )}
- {onAssignSpool && (
- <button
- onClick={(e) => { e.stopPropagation(); onAssignSpool(); }}
- className="w-full flex items-center justify-center gap-1.5 px-2 py-1.5 text-xs font-medium rounded transition-colors bg-bambu-blue/20 hover:bg-bambu-blue/30 text-bambu-blue"
- >
- <Package className="w-3.5 h-3.5" />
- {t('inventory.assignSpool')}
- </button>
- )}
- </div>
- )}
- </div>
- <div className="
- absolute left-1/2 -translate-x-1/2 top-full w-0 h-0
- border-l-[5px] border-l-transparent
- border-r-[5px] border-r-transparent
- border-t-[5px] border-t-bambu-dark-tertiary
- " />
- </div>,
- document.body,
- )}
- </div>
- );
- }
|