Bin
2025-12-17 2b99d77d73ba568beff0a549534017caaad8a6de
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
import React, { useCallback, useMemo } from "react";
import clsx from "clsx";
import { cn } from "../../../utils/bem";
import { useDropdown } from "@humansignal/ui";
import "./Menu.scss";
import { MenuContext } from "./MenuContext";
import { MenuItem } from "./MenuItem";
 
const menuCN = cn("menu-dm");
 
export const Menu = React.forwardRef(
  ({ children, className, style, size, selectedKeys, closeDropdownOnItemClick }, ref) => {
    const dropdown = useDropdown();
 
    const selected = useMemo(() => {
      return new Set(selectedKeys ?? []);
    }, [selectedKeys]);
 
    const clickHandler = useCallback(
      (e) => {
        const elem = menuCN.elem("item").closest(e.target);
 
        if (dropdown && elem && closeDropdownOnItemClick !== false) {
          dropdown.close();
        }
      },
      [dropdown],
    );
 
    const collapsed = useMemo(() => {
      return !!dropdown;
    }, [dropdown]);
 
    return (
      <MenuContext.Provider value={{ selected }}>
        <ul
          ref={ref}
          className={clsx(menuCN.toString(), menuCN.mod({ size, collapsed }).toString(), className)}
          style={style}
          onClick={clickHandler}
        >
          {children}
        </ul>
      </MenuContext.Provider>
    );
  },
);
 
Menu.Item = MenuItem;
Menu.Spacer = () => <li className={cn("menu-dm", { elem: "spacer" })} />;
Menu.Divider = () => <li className={cn("menu-dm", { elem: "divider" })} />;
Menu.Builder = (url, menuItems) => {
  return (menuItems ?? []).map((item, index) => {
    if (item === "SPACER") return <Menu.Spacer key={index} />;
    if (item === "DIVIDER") return <Menu.Divider key={index} />;
 
    const [path, label] = item;
    const location = `${url}${path}`.replace(/([/]+)/g, "/");
 
    return (
      <Menu.Item key={index} to={location} exact>
        {label}
      </Menu.Item>
    );
  });
};
 
Menu.Group = ({ children, title, className, style }) => {
  const rootClass = cn("menu-group-dm");
 
  return (
    <li className={rootClass.mix(className)} style={style}>
      <div className={rootClass.elem("title")}>{title}</div>
      <ul className={rootClass.elem("list")}>{children}</ul>
    </li>
  );
};