import { useEffect, useMemo } from 'react'; import { useRightRail } from '../contexts/RightRailContext'; import { RightRailAction, RightRailButtonConfig } from '../types/rightRail'; export interface RightRailButtonWithAction extends RightRailButtonConfig { onClick: RightRailAction; } /** * Registers one or more RightRail buttons and their actions. * - Automatically registers on mount and unregisters on unmount * - Updates registration when the input array reference changes */ export function useRightRailButtons(buttons: readonly RightRailButtonWithAction[]) { const { registerButtons, unregisterButtons, setAction } = useRightRail(); // Memoize configs and ids to reduce churn const configs: RightRailButtonConfig[] = useMemo( () => buttons.map(({ onClick, ...cfg }) => cfg), [buttons] ); const ids: string[] = useMemo(() => buttons.map(b => b.id), [buttons]); useEffect(() => { if (!buttons || buttons.length === 0) return; // DEV warnings for duplicate ids or missing handlers if (process.env.NODE_ENV === 'development') { const idSet = new Set(); buttons.forEach(b => { if (!b.onClick) console.warn('[RightRail] Missing onClick for id:', b.id); if (idSet.has(b.id)) console.warn('[RightRail] Duplicate id in buttons array:', b.id); idSet.add(b.id); }); } // Register visual button configs (idempotent merge by id) registerButtons(configs); // Bind/update actions independent of registration buttons.forEach(({ id, onClick }) => setAction(id, onClick)); // Cleanup unregisters by ids present in this call return () => unregisterButtons(ids); }, [registerButtons, unregisterButtons, setAction, configs, ids, buttons]); }