ContextMenu.tsx 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303
  1. import { useEffect, useRef, useState, useLayoutEffect } from 'react';
  2. import { ChevronRight, Search } from 'lucide-react';
  3. export interface ContextMenuItem {
  4. label: string;
  5. icon?: React.ReactNode;
  6. onClick: () => void;
  7. danger?: boolean;
  8. disabled?: boolean;
  9. divider?: boolean;
  10. submenu?: ContextMenuItem[];
  11. // When set on an item with a submenu, render a search input above the
  12. // submenu items that filters by label (case-insensitive).
  13. submenuSearchPlaceholder?: string;
  14. title?: string;
  15. }
  16. interface ContextMenuProps {
  17. x: number;
  18. y: number;
  19. items: ContextMenuItem[];
  20. onClose: () => void;
  21. }
  22. interface SubmenuPanelProps {
  23. items: ContextMenuItem[];
  24. searchPlaceholder?: string;
  25. onClose: () => void;
  26. className: string;
  27. onMouseEnter: () => void;
  28. onMouseLeave: () => void;
  29. }
  30. function SubmenuPanel({
  31. items,
  32. searchPlaceholder,
  33. onClose,
  34. className,
  35. onMouseEnter,
  36. onMouseLeave,
  37. }: SubmenuPanelProps) {
  38. const [query, setQuery] = useState('');
  39. const inputRef = useRef<HTMLInputElement>(null);
  40. useEffect(() => {
  41. if (searchPlaceholder) {
  42. // Defer focus so it survives the mouse event that opened the submenu.
  43. const id = window.setTimeout(() => inputRef.current?.focus(), 0);
  44. return () => window.clearTimeout(id);
  45. }
  46. }, [searchPlaceholder]);
  47. const trimmed = query.trim().toLowerCase();
  48. const filteredItems = searchPlaceholder && trimmed
  49. ? items.filter((i) => i.label.toLowerCase().includes(trimmed))
  50. : items;
  51. return (
  52. <div
  53. className={className}
  54. onMouseEnter={onMouseEnter}
  55. onMouseLeave={onMouseLeave}
  56. >
  57. {searchPlaceholder && (
  58. <div className="sticky top-0 z-[1] px-2 py-1.5 bg-bambu-dark-secondary border-b border-bambu-dark-tertiary">
  59. <div className="relative">
  60. <Search className="absolute left-2 top-1/2 -translate-y-1/2 w-3.5 h-3.5 text-bambu-gray pointer-events-none" />
  61. <input
  62. ref={inputRef}
  63. type="text"
  64. value={query}
  65. onChange={(e) => setQuery(e.target.value)}
  66. onKeyDown={(e) => {
  67. if (e.key === 'Enter') {
  68. const first = filteredItems.find((i) => !i.disabled);
  69. if (first) {
  70. first.onClick();
  71. onClose();
  72. }
  73. }
  74. }}
  75. placeholder={searchPlaceholder}
  76. className="w-full pl-7 pr-2 py-1 text-sm bg-bambu-dark border border-bambu-dark-tertiary rounded text-white placeholder-bambu-gray focus:border-bambu-green focus:outline-none"
  77. />
  78. </div>
  79. </div>
  80. )}
  81. {filteredItems.map((subItem, subIndex) => (
  82. <button
  83. key={subIndex}
  84. onClick={() => {
  85. if (!subItem.disabled) {
  86. subItem.onClick();
  87. onClose();
  88. }
  89. }}
  90. disabled={subItem.disabled}
  91. className={`w-full flex items-center gap-2 px-3 py-2 text-sm text-left transition-colors ${
  92. subItem.disabled
  93. ? 'text-bambu-gray cursor-not-allowed'
  94. : subItem.danger
  95. ? 'text-red-400 hover:bg-red-400/10'
  96. : 'text-white hover:bg-bambu-dark-tertiary'
  97. }`}
  98. >
  99. {subItem.icon && <span className="w-4 h-4 flex-shrink-0 flex items-center justify-center">{subItem.icon}</span>}
  100. <span className="flex-1 truncate">{subItem.label}</span>
  101. </button>
  102. ))}
  103. {searchPlaceholder && filteredItems.length === 0 && (
  104. <div className="px-3 py-2 text-sm text-bambu-gray text-center italic">—</div>
  105. )}
  106. </div>
  107. );
  108. }
  109. export function ContextMenu({ x, y, items, onClose }: ContextMenuProps) {
  110. const menuRef = useRef<HTMLDivElement>(null);
  111. const [activeSubmenu, setActiveSubmenu] = useState<number | null>(null);
  112. const submenuTimeoutRef = useRef<number | null>(null);
  113. const [position, setPosition] = useState({ x, y, visible: false });
  114. const [openSubmenuLeft, setOpenSubmenuLeft] = useState(false);
  115. const [submenuPositions, setSubmenuPositions] = useState<Record<number, 'top' | 'bottom'>>({});
  116. useEffect(() => {
  117. const handleClickOutside = (e: MouseEvent) => {
  118. if (menuRef.current && !menuRef.current.contains(e.target as Node)) {
  119. onClose();
  120. }
  121. };
  122. const handleEscape = (e: KeyboardEvent) => {
  123. if (e.key === 'Escape') {
  124. onClose();
  125. }
  126. };
  127. const handleScroll = (e: Event) => {
  128. // Internal submenu scroll (overflow-y-auto on the submenu panel) must
  129. // not dismiss the menu — only close on scroll outside our own subtree.
  130. if (menuRef.current && menuRef.current.contains(e.target as Node)) {
  131. return;
  132. }
  133. onClose();
  134. };
  135. document.addEventListener('mousedown', handleClickOutside);
  136. document.addEventListener('keydown', handleEscape);
  137. document.addEventListener('scroll', handleScroll, true);
  138. return () => {
  139. document.removeEventListener('mousedown', handleClickOutside);
  140. document.removeEventListener('keydown', handleEscape);
  141. document.removeEventListener('scroll', handleScroll, true);
  142. if (submenuTimeoutRef.current) {
  143. clearTimeout(submenuTimeoutRef.current);
  144. }
  145. };
  146. }, [onClose]);
  147. // Adjust position to keep menu in viewport - use useLayoutEffect for synchronous measurement
  148. useLayoutEffect(() => {
  149. if (menuRef.current) {
  150. // Force a reflow to get accurate measurements
  151. menuRef.current.style.visibility = 'hidden';
  152. menuRef.current.style.display = 'block';
  153. const rect = menuRef.current.getBoundingClientRect();
  154. const viewportWidth = window.innerWidth;
  155. const viewportHeight = window.innerHeight;
  156. const padding = 8;
  157. let adjustedX = x;
  158. let adjustedY = y;
  159. // Adjust horizontal position - if menu would overflow right, shift left
  160. if (x + rect.width > viewportWidth - padding) {
  161. adjustedX = Math.max(padding, viewportWidth - rect.width - padding);
  162. }
  163. // Also check if starting position is negative
  164. if (adjustedX < padding) {
  165. adjustedX = padding;
  166. }
  167. // Adjust vertical position - if menu would overflow bottom, shift up
  168. if (y + rect.height > viewportHeight - padding) {
  169. adjustedY = Math.max(padding, viewportHeight - rect.height - padding);
  170. }
  171. // Also check if starting position is negative
  172. if (adjustedY < padding) {
  173. adjustedY = padding;
  174. }
  175. // Check if submenus should open to the left (more space on left than right)
  176. const submenuWidth = 180;
  177. const spaceOnRight = viewportWidth - adjustedX - rect.width;
  178. const spaceOnLeft = adjustedX;
  179. // Only open left if there's not enough space on right AND there's enough space on left
  180. setOpenSubmenuLeft(spaceOnRight < submenuWidth && spaceOnLeft > submenuWidth);
  181. setPosition({ x: adjustedX, y: adjustedY, visible: true });
  182. }
  183. }, [x, y]);
  184. const handleMouseEnterSubmenu = (index: number, element: HTMLElement) => {
  185. if (submenuTimeoutRef.current) {
  186. clearTimeout(submenuTimeoutRef.current);
  187. submenuTimeoutRef.current = null;
  188. }
  189. // Calculate if submenu should open upward or downward
  190. const rect = element.getBoundingClientRect();
  191. const viewportHeight = window.innerHeight;
  192. const submenuMaxHeight = 300; // matches max-h-[300px]
  193. const padding = 8;
  194. // Check if there's enough space below for the submenu
  195. const spaceBelow = viewportHeight - rect.top - padding;
  196. const shouldOpenUpward = spaceBelow < submenuMaxHeight && rect.top > submenuMaxHeight;
  197. setSubmenuPositions(prev => ({ ...prev, [index]: shouldOpenUpward ? 'bottom' : 'top' }));
  198. setActiveSubmenu(index);
  199. };
  200. const handleMouseLeaveSubmenu = () => {
  201. submenuTimeoutRef.current = window.setTimeout(() => {
  202. setActiveSubmenu(null);
  203. }, 150);
  204. };
  205. return (
  206. <div
  207. ref={menuRef}
  208. className="fixed z-50 min-w-[180px] max-w-[280px] bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg shadow-xl py-1"
  209. style={{
  210. left: position.x,
  211. top: position.y,
  212. visibility: position.visible ? 'visible' : 'hidden'
  213. }}
  214. >
  215. {items.map((item, index) => {
  216. if (item.divider) {
  217. return <div key={index} className="my-1 border-t border-bambu-dark-tertiary" />;
  218. }
  219. const hasSubmenu = item.submenu && item.submenu.length > 0;
  220. return (
  221. <div
  222. key={index}
  223. className="relative"
  224. onMouseEnter={(e) => hasSubmenu && handleMouseEnterSubmenu(index, e.currentTarget)}
  225. onMouseLeave={() => hasSubmenu && handleMouseLeaveSubmenu()}
  226. >
  227. <button
  228. onMouseEnter={(e) => hasSubmenu && handleMouseEnterSubmenu(index, e.currentTarget.parentElement!)}
  229. onClick={() => {
  230. if (hasSubmenu) {
  231. // Toggle submenu on click as well
  232. setActiveSubmenu(activeSubmenu === index ? null : index);
  233. } else if (!item.disabled) {
  234. item.onClick();
  235. onClose();
  236. }
  237. }}
  238. disabled={item.disabled}
  239. title={item.title}
  240. className={`w-full flex items-center gap-2 px-3 py-2 text-sm text-left transition-colors ${
  241. item.disabled
  242. ? 'text-bambu-gray cursor-not-allowed'
  243. : item.danger
  244. ? 'text-red-400 hover:bg-red-400/10'
  245. : 'text-white hover:bg-bambu-dark-tertiary'
  246. } ${hasSubmenu && activeSubmenu === index ? 'bg-bambu-dark-tertiary' : ''}`}
  247. >
  248. {item.icon && <span className="w-4 h-4 flex-shrink-0 flex items-center justify-center">{item.icon}</span>}
  249. <span className="flex-1">{item.label}</span>
  250. {hasSubmenu && <ChevronRight className="w-4 h-4 text-bambu-gray" />}
  251. </button>
  252. {/* Submenu */}
  253. {hasSubmenu && activeSubmenu === index && (
  254. <SubmenuPanel
  255. items={item.submenu!}
  256. searchPlaceholder={item.submenuSearchPlaceholder}
  257. onClose={onClose}
  258. className={`absolute min-w-[200px] bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg shadow-xl py-1 overflow-hidden max-h-[300px] overflow-y-auto z-[60] ${
  259. openSubmenuLeft ? 'right-full mr-1' : 'left-full ml-1'
  260. } ${submenuPositions[index] === 'bottom' ? 'bottom-0' : 'top-0'}`}
  261. onMouseEnter={() => {
  262. if (submenuTimeoutRef.current) {
  263. clearTimeout(submenuTimeoutRef.current);
  264. submenuTimeoutRef.current = null;
  265. }
  266. }}
  267. onMouseLeave={() => handleMouseLeaveSubmenu()}
  268. />
  269. )}
  270. </div>
  271. );
  272. })}
  273. </div>
  274. );
  275. }