summaryrefslogtreecommitdiff
path: root/makima/frontend/src/components/SidebarContextMenu.tsx
blob: a7b0ae6a9a33ac70dc903e000b37218505541991 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
import { useEffect, useRef } from "react";

// Generic right-click context menu for the sidebar tree. Each call site
// (directive folder, contract row, step row, task row, …) builds its own
// items array. Lifts the viewport-clamping + click-outside / Esc logic
// out of the deleted DirectiveContextMenu so we don't end up with one
// component per entity type.

export interface ContextMenuItem {
  /** Display text. Empty for separators. */
  label: string;
  /** Click handler. Ignored when `separator` is true. */
  onClick?: () => void;
  /** Render in red — for destructive operations (delete, archive). */
  danger?: boolean;
  /** Render as a thin divider instead of a button. */
  separator?: boolean;
  /** Greyed out, non-clickable. Used for items whose preconditions
   *  aren't met (e.g. "Reopen" on an already-draft contract). */
  disabled?: boolean;
}

interface SidebarContextMenuProps {
  x: number;
  y: number;
  items: ContextMenuItem[];
  onClose: () => void;
}

export function SidebarContextMenu({ x, y, items, onClose }: SidebarContextMenuProps) {
  const menuRef = useRef<HTMLDivElement>(null);

  // Close on click outside or Esc.
  useEffect(() => {
    const handleClickOutside = (e: MouseEvent) => {
      if (menuRef.current && !menuRef.current.contains(e.target as Node)) {
        onClose();
      }
    };
    const handleKeyDown = (e: KeyboardEvent) => {
      if (e.key === "Escape") onClose();
    };
    document.addEventListener("mousedown", handleClickOutside);
    document.addEventListener("keydown", handleKeyDown);
    return () => {
      document.removeEventListener("mousedown", handleClickOutside);
      document.removeEventListener("keydown", handleKeyDown);
    };
  }, [onClose]);

  // Clamp into viewport if the menu would overflow off the right or
  // bottom edge.
  useEffect(() => {
    if (menuRef.current) {
      const rect = menuRef.current.getBoundingClientRect();
      const viewportWidth = window.innerWidth;
      const viewportHeight = window.innerHeight;
      if (rect.right > viewportWidth) {
        menuRef.current.style.left = `${x - rect.width}px`;
      }
      if (rect.bottom > viewportHeight) {
        menuRef.current.style.top = `${y - rect.height}px`;
      }
    }
  }, [x, y]);

  const baseItemClass =
    "w-full px-3 py-1.5 text-left text-xs font-mono flex items-center gap-2";
  const enabledClass = "text-[#9bc3ff] hover:bg-[rgba(117,170,252,0.1)]";
  const dangerClass = "text-red-400 hover:bg-[rgba(239,68,68,0.1)]";
  const disabledClass = "text-[#3a4a6a] cursor-not-allowed";
  const dividerClass = "border-t border-[rgba(117,170,252,0.2)] my-1";

  return (
    <div
      ref={menuRef}
      className="fixed z-50 min-w-[180px] bg-[#0a1628] border border-[rgba(117,170,252,0.3)] rounded shadow-lg py-1"
      style={{ left: x, top: y }}
      onContextMenu={(e) => e.preventDefault()}
    >
      {items.map((item, i) => {
        if (item.separator) {
          return <div key={`sep-${i}`} className={dividerClass} />;
        }
        const cls = item.disabled
          ? disabledClass
          : item.danger
          ? dangerClass
          : enabledClass;
        return (
          <button
            key={`item-${i}-${item.label}`}
            type="button"
            disabled={item.disabled}
            onClick={() => {
              if (item.disabled) return;
              item.onClick?.();
              onClose();
            }}
            className={`${baseItemClass} ${cls}`}
          >
            {item.label}
          </button>
        );
      })}
    </div>
  );
}