@ultraviolet/ui 1.87.1 → 1.87.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -3,6 +3,16 @@ Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
3
3
  const jsxRuntime = require("@emotion/react/jsx-runtime");
4
4
  const index = require("../Stack/index.cjs");
5
5
  const index$1 = require("../Text/index.cjs");
6
+ const LabelRequiredOrNot = ({
7
+ children,
8
+ required,
9
+ id,
10
+ size,
11
+ htmlFor
12
+ }) => required ? /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { direction: "row", gap: "0.5", alignItems: "start", children: [
13
+ /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "label", id, variant: size === "large" ? "bodyStrong" : "bodySmallStrong", sentiment: "neutral", htmlFor, prominence: "strong", children }),
14
+ /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "span", variant: size === "large" ? "bodyStrong" : "bodySmallStrong", sentiment: "danger", "aria-label": "required", children: "*" })
15
+ ] }) : /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "label", id, variant: size === "large" ? "bodyStrong" : "bodySmallStrong", sentiment: "neutral", htmlFor, prominence: "strong", children });
6
16
  const Label = ({
7
17
  children,
8
18
  labelDescription,
@@ -10,14 +20,8 @@ const Label = ({
10
20
  size = "large",
11
21
  htmlFor,
12
22
  id
13
- }) => {
14
- const LabelRequiredOrNot = () => required ? /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { direction: "row", gap: "0.5", alignItems: "start", children: [
15
- /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "label", id, variant: size === "large" ? "bodyStrong" : "bodySmallStrong", sentiment: "neutral", htmlFor, prominence: "strong", children }),
16
- /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "span", variant: size === "large" ? "bodyStrong" : "bodySmallStrong", sentiment: "danger", "aria-label": "required", children: "*" })
17
- ] }) : /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "label", id, variant: size === "large" ? "bodyStrong" : "bodySmallStrong", sentiment: "neutral", htmlFor, prominence: "strong", children });
18
- return labelDescription ? /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { direction: "row", gap: "1", alignItems: "center", children: [
19
- /* @__PURE__ */ jsxRuntime.jsx(LabelRequiredOrNot, {}),
20
- typeof labelDescription === "string" ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "span", variant: "bodySmall", children: labelDescription }) : labelDescription
21
- ] }) : /* @__PURE__ */ jsxRuntime.jsx(LabelRequiredOrNot, {});
22
- };
23
+ }) => labelDescription ? /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { direction: "row", gap: "1", alignItems: "center", children: [
24
+ /* @__PURE__ */ jsxRuntime.jsx(LabelRequiredOrNot, { required, size, htmlFor, id, children }),
25
+ typeof labelDescription === "string" ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "span", variant: "bodySmall", children: labelDescription }) : labelDescription
26
+ ] }) : /* @__PURE__ */ jsxRuntime.jsx(LabelRequiredOrNot, { required, size, htmlFor, id, children });
23
27
  exports.Label = Label;
@@ -1,6 +1,16 @@
1
1
  import { jsxs, jsx } from "@emotion/react/jsx-runtime";
2
2
  import { Stack } from "../Stack/index.js";
3
3
  import { Text } from "../Text/index.js";
4
+ const LabelRequiredOrNot = ({
5
+ children,
6
+ required,
7
+ id,
8
+ size,
9
+ htmlFor
10
+ }) => required ? /* @__PURE__ */ jsxs(Stack, { direction: "row", gap: "0.5", alignItems: "start", children: [
11
+ /* @__PURE__ */ jsx(Text, { as: "label", id, variant: size === "large" ? "bodyStrong" : "bodySmallStrong", sentiment: "neutral", htmlFor, prominence: "strong", children }),
12
+ /* @__PURE__ */ jsx(Text, { as: "span", variant: size === "large" ? "bodyStrong" : "bodySmallStrong", sentiment: "danger", "aria-label": "required", children: "*" })
13
+ ] }) : /* @__PURE__ */ jsx(Text, { as: "label", id, variant: size === "large" ? "bodyStrong" : "bodySmallStrong", sentiment: "neutral", htmlFor, prominence: "strong", children });
4
14
  const Label = ({
5
15
  children,
6
16
  labelDescription,
@@ -8,16 +18,10 @@ const Label = ({
8
18
  size = "large",
9
19
  htmlFor,
10
20
  id
11
- }) => {
12
- const LabelRequiredOrNot = () => required ? /* @__PURE__ */ jsxs(Stack, { direction: "row", gap: "0.5", alignItems: "start", children: [
13
- /* @__PURE__ */ jsx(Text, { as: "label", id, variant: size === "large" ? "bodyStrong" : "bodySmallStrong", sentiment: "neutral", htmlFor, prominence: "strong", children }),
14
- /* @__PURE__ */ jsx(Text, { as: "span", variant: size === "large" ? "bodyStrong" : "bodySmallStrong", sentiment: "danger", "aria-label": "required", children: "*" })
15
- ] }) : /* @__PURE__ */ jsx(Text, { as: "label", id, variant: size === "large" ? "bodyStrong" : "bodySmallStrong", sentiment: "neutral", htmlFor, prominence: "strong", children });
16
- return labelDescription ? /* @__PURE__ */ jsxs(Stack, { direction: "row", gap: "1", alignItems: "center", children: [
17
- /* @__PURE__ */ jsx(LabelRequiredOrNot, {}),
18
- typeof labelDescription === "string" ? /* @__PURE__ */ jsx(Text, { as: "span", variant: "bodySmall", children: labelDescription }) : labelDescription
19
- ] }) : /* @__PURE__ */ jsx(LabelRequiredOrNot, {});
20
- };
21
+ }) => labelDescription ? /* @__PURE__ */ jsxs(Stack, { direction: "row", gap: "1", alignItems: "center", children: [
22
+ /* @__PURE__ */ jsx(LabelRequiredOrNot, { required, size, htmlFor, id, children }),
23
+ typeof labelDescription === "string" ? /* @__PURE__ */ jsx(Text, { as: "span", variant: "bodySmall", children: labelDescription }) : labelDescription
24
+ ] }) : /* @__PURE__ */ jsx(LabelRequiredOrNot, { required, size, htmlFor, id, children });
21
25
  export {
22
26
  Label
23
27
  };
@@ -24,13 +24,13 @@ const StyledPopup = /* @__PURE__ */ _styled__default.default(index.Popup, proces
24
24
  theme
25
25
  }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`, ";padding:0;&[data-has-arrow='true']{&::after{border-color:", ({
26
26
  theme
27
- }) => theme.colors.other.elevation.background.raised, " transparent transparent transparent;}}width:", ({
27
+ }) => theme.colors.other.elevation.background.raised, " transparent transparent transparent;}}min-width:", constants.SIZES.small, ";max-width:", constants.SIZES.large, ";", ({
28
28
  size
29
- }) => constants.SIZES[size], ";max-width:none;", ({
29
+ }) => size ? `width: ${constants.SIZES[size]};` : null, " ", ({
30
30
  searchable
31
31
  }) => searchable ? `min-width: 20rem` : null, ";padding:", ({
32
32
  theme
33
- }) => `${theme.space["0.25"]} 0`, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01lbnVWMi9NZW51Q29udGVudC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBd0JxRCIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9NZW51VjIvTWVudUNvbnRlbnQudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7IEJ1dHRvbkhUTUxBdHRyaWJ1dGVzLCBNb3VzZUV2ZW50LCBSZWFjdE5vZGUsIFJlZiB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHtcbiAgY2xvbmVFbGVtZW50LFxuICBmb3J3YXJkUmVmLFxuICBpc1ZhbGlkRWxlbWVudCxcbiAgdXNlQ2FsbGJhY2ssXG4gIHVzZUVmZmVjdCxcbiAgdXNlSWQsXG4gIHVzZUltcGVyYXRpdmVIYW5kbGUsXG4gIHVzZU1lbW8sXG4gIHVzZVJlZixcbiAgdXNlU3RhdGUsXG59IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgUG9wdXAgfSBmcm9tICcuLi9Qb3B1cCdcbmltcG9ydCB7IFNlYXJjaElucHV0IH0gZnJvbSAnLi4vU2VhcmNoSW5wdXQnXG5pbXBvcnQgeyBTdGFjayB9IGZyb20gJy4uL1N0YWNrJ1xuaW1wb3J0IHsgdXNlTWVudSB9IGZyb20gJy4vTWVudVByb3ZpZGVyJ1xuaW1wb3J0IHsgU0laRVMgfSBmcm9tICcuL2NvbnN0YW50cydcbmltcG9ydCB7IHNlYXJjaENoaWxkcmVuIH0gZnJvbSAnLi9oZWxwZXJzJ1xuaW1wb3J0IHR5cGUgeyBNZW51UHJvcHMgfSBmcm9tICcuL3R5cGVzJ1xuXG5jb25zdCBTdHlsZWRQb3B1cCA9IHN0eWxlZChQb3B1cCwge1xuICBzaG91bGRGb3J3YXJkUHJvcDogcHJvcCA9PiAhWydzaXplJywgJ3NlYXJjaGFibGUnXS5pbmNsdWRlcyhwcm9wKSxcbn0pPHsgc2l6ZToga2V5b2YgdHlwZW9mIFNJWkVTOyBzZWFyY2hhYmxlOiBib29sZWFuIH0+YFxuICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLmNvbG9ycy5vdGhlci5lbGV2YXRpb24uYmFja2dyb3VuZC5yYWlzZWR9O1xuICBib3gtc2hhZG93OiAkeyh7IHRoZW1lIH0pID0+IGAke3RoZW1lLnNoYWRvd3MucmFpc2VkWzBdfSwgJHt0aGVtZS5zaGFkb3dzLnJhaXNlZFsxXX1gfTtcbiAgcGFkZGluZzogMDtcblxuICAmW2RhdGEtaGFzLWFycm93PSd0cnVlJ10ge1xuICAgICY6OmFmdGVyIHtcbiAgICAgIGJvcmRlci1jb2xvcjogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5jb2xvcnMub3RoZXIuZWxldmF0aW9uLmJhY2tncm91bmQucmFpc2VkfVxuICAgICAgICB0cmFuc3BhcmVudCB0cmFuc3BhcmVudCB0cmFuc3BhcmVudDtcbiAgICB9XG4gIH1cblxuICB3aWR0aDogJHsoeyBzaXplIH0pID0+IFNJWkVTW3NpemVdfTtcbiAgbWF4LXdpZHRoOiBub25lO1xuICAkeyh7IHNlYXJjaGFibGUgfSkgPT4gKHNlYXJjaGFibGUgPyBgbWluLXdpZHRoOiAyMHJlbWAgOiBudWxsKX07XG4gIHBhZGRpbmc6ICR7KHsgdGhlbWUgfSkgPT4gYCR7dGhlbWUuc3BhY2VbJzAuMjUnXX0gMGB9O1xuYFxuXG5jb25zdCBNZW51TGlzdCA9IHN0eWxlZChTdGFjaylgXG4gIG92ZXJmbG93LXk6IGF1dG87XG4gIG92ZXJmbG93LXg6IGhpZGRlbjtcbiAgJjphZnRlcixcbiAgJjpiZWZvcmUge1xuICAgIGJvcmRlcjogc29saWQgdHJhbnNwYXJlbnQ7XG4gICAgYm9yZGVyLXdpZHRoOiA5cHg7XG4gICAgY29udGVudDogJyAnO1xuICAgIGhlaWdodDogMDtcbiAgICB3aWR0aDogMDtcbiAgICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gICAgcG9pbnRlci1ldmVudHM6IG5vbmU7XG4gIH1cblxuICAmOmFmdGVyIHtcbiAgICBib3JkZXItY29sb3I6IHRyYW5zcGFyZW50O1xuICB9XG4gICY6YmVmb3JlIHtcbiAgICBib3JkZXItY29sb3I6IHRyYW5zcGFyZW50O1xuICB9XG4gIGJhY2tncm91bmQtY29sb3I6ICR7KHsgdGhlbWUgfSkgPT5cbiAgICB0aGVtZS5jb2xvcnMub3RoZXIuZWxldmF0aW9uLmJhY2tncm91bmQucmFpc2VkfTtcbiAgY29sb3I6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuY29sb3JzLm5ldXRyYWwudGV4dH07XG4gIGJvcmRlci1yYWRpdXM6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUucmFkaWkuZGVmYXVsdH07XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbmBcblxuY29uc3QgU3R5bGVkU2VhcmNoSW5wdXQgPSBzdHlsZWQoU2VhcmNoSW5wdXQpYFxuICBwYWRkaW5nOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLnNwYWNlWycxJ119O1xuYFxuXG5leHBvcnQgY29uc3QgTWVudSA9IGZvcndhcmRSZWYoXG4gIChcbiAgICB7XG4gICAgICBpZCxcbiAgICAgIGFyaWFMYWJlbCA9ICdNZW51JyxcbiAgICAgIGNoaWxkcmVuLFxuICAgICAgZGlzY2xvc3VyZSxcbiAgICAgIGhhc0Fycm93ID0gZmFsc2UsXG4gICAgICBwbGFjZW1lbnQgPSAnYm90dG9tJyxcbiAgICAgIGNsYXNzTmFtZSxcbiAgICAgICdkYXRhLXRlc3RpZCc6IGRhdGFUZXN0SWQsXG4gICAgICBtYXhIZWlnaHQsXG4gICAgICBtYXhXaWR0aCxcbiAgICAgIHBvcnRhbFRhcmdldCA9IGRvY3VtZW50LmJvZHksXG4gICAgICBzaXplID0gJ3NtYWxsJyxcbiAgICAgIHRyaWdnZXJNZXRob2QgPSAnY2xpY2snLFxuICAgICAgZHluYW1pY0RvbVJlbmRlcmluZyxcbiAgICAgIGFsaWduLFxuICAgICAgc2VhcmNoYWJsZSA9IGZhbHNlLFxuICAgIH06IE1lbnVQcm9wcyxcbiAgICByZWY6IFJlZjxIVE1MQnV0dG9uRWxlbWVudCB8IG51bGw+LFxuICApID0+IHtcbiAgICBjb25zdCB7IGlzVmlzaWJsZSwgc2V0SXNWaXNpYmxlIH0gPSB1c2VNZW51KClcbiAgICBjb25zdCBzZWFyY2hJbnB1dFJlZiA9IHVzZVJlZjxIVE1MSW5wdXRFbGVtZW50PihudWxsKVxuICAgIGNvbnN0IFtsb2NhbENoaWxkLCBzZXRMb2NhbENoaWxkXSA9IHVzZVN0YXRlPFJlYWN0Tm9kZVtdIHwgbnVsbD4obnVsbClcbiAgICBjb25zdCBwb3B1cFJlZiA9IHVzZVJlZjxIVE1MRGl2RWxlbWVudD4obnVsbClcbiAgICBjb25zdCBkaXNjbG9zdXJlUmVmID0gdXNlUmVmPEhUTUxCdXR0b25FbGVtZW50PihudWxsKVxuICAgIGNvbnN0IHRlbXBJZCA9IHVzZUlkKClcbiAgICBjb25zdCBmaW5hbElkID0gYG1lbnUtJHtpZCA/PyB0ZW1wSWR9YFxuXG4gICAgLy8gaWYgeW91IG5lZWQgZGlhbG9nIGluc2lkZSB5b3VyIGNvbXBvbmVudCwgdXNlIGZ1bmN0aW9uLCBvdGhlcndpc2UgY29tcG9uZW50IGlzIGZpbmVcbiAgICBjb25zdCB0YXJnZXQgPSBpc1ZhbGlkRWxlbWVudDxCdXR0b25IVE1MQXR0cmlidXRlczxIVE1MQnV0dG9uRWxlbWVudD4+KFxuICAgICAgZGlzY2xvc3VyZSxcbiAgICApXG4gICAgICA/IGRpc2Nsb3N1cmVcbiAgICAgIDogZGlzY2xvc3VyZSh7IHZpc2libGU6IGlzVmlzaWJsZSB9KVxuICAgIGNvbnN0IGlubmVyUmVmID0gdXNlUmVmKHRhcmdldCBhcyB1bmtub3duIGFzIEhUTUxCdXR0b25FbGVtZW50KVxuICAgIHVzZUltcGVyYXRpdmVIYW5kbGUocmVmLCAoKSA9PiBpbm5lclJlZi5jdXJyZW50KVxuXG4gICAgY29uc3QgZmluYWxEaXNjbG9zdXJlID0gY2xvbmVFbGVtZW50KHRhcmdldCwge1xuICAgICAgb25DbGljazogKGV2ZW50OiBNb3VzZUV2ZW50PEhUTUxCdXR0b25FbGVtZW50PikgPT4ge1xuICAgICAgICB0YXJnZXQucHJvcHMub25DbGljaz8uKGV2ZW50KVxuICAgICAgICBzZXRJc1Zpc2libGUoIWlzVmlzaWJsZSlcbiAgICAgIH0sXG4gICAgICAnYXJpYS1oYXNwb3B1cCc6ICdkaWFsb2cnLFxuICAgICAgJ2FyaWEtZXhwYW5kZWQnOiBpc1Zpc2libGUsXG4gICAgICAvLyBAdHMtZXhwZWN0LWVycm9yIG5vdCBzdXJlIGhvdyB0byBmaXggdGhpc1xuICAgICAgcmVmOiBkaXNjbG9zdXJlUmVmLFxuICAgIH0pXG5cbiAgICBjb25zdCBvblNlYXJjaCA9IHVzZUNhbGxiYWNrKFxuICAgICAgKHZhbHVlOiBzdHJpbmcpID0+IHtcbiAgICAgICAgaWYgKHR5cGVvZiBjaGlsZHJlbiA9PT0gJ29iamVjdCcpIHtcbiAgICAgICAgICBzZXRMb2NhbENoaWxkKHNlYXJjaENoaWxkcmVuKGNoaWxkcmVuLCB2YWx1ZSkpXG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgICBbY2hpbGRyZW5dLFxuICAgIClcblxuICAgIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgICBpZiAoaXNWaXNpYmxlICYmIHNlYXJjaGFibGUpIHtcbiAgICAgICAgc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICAgICAgc2VhcmNoSW5wdXRSZWYuY3VycmVudD8uZm9jdXMoKVxuICAgICAgICB9LCA1MClcbiAgICAgIH1cbiAgICB9LCBbaXNWaXNpYmxlLCBzZWFyY2hhYmxlXSlcblxuICAgIGNvbnN0IGZpbmFsQ2hpbGQgPSB1c2VNZW1vKCgpID0+IHtcbiAgICAgIGlmICh0eXBlb2YgY2hpbGRyZW4gPT09ICdmdW5jdGlvbicpIHtcbiAgICAgICAgcmV0dXJuIGNoaWxkcmVuKHsgdG9nZ2xlOiAoKSA9PiBzZXRJc1Zpc2libGUoIWlzVmlzaWJsZSkgfSlcbiAgICAgIH1cblxuICAgICAgaWYgKHNlYXJjaGFibGUgJiYgbG9jYWxDaGlsZCkge1xuICAgICAgICByZXR1cm4gbG9jYWxDaGlsZFxuICAgICAgfVxuXG4gICAgICByZXR1cm4gY2hpbGRyZW5cbiAgICB9LCBbY2hpbGRyZW4sIGlzVmlzaWJsZSwgbG9jYWxDaGlsZCwgc2VhcmNoYWJsZSwgc2V0SXNWaXNpYmxlXSlcblxuICAgIHJldHVybiAoXG4gICAgICA8U3R5bGVkUG9wdXBcbiAgICAgICAgZGVib3VuY2VEZWxheT17dHJpZ2dlck1ldGhvZCA9PT0gJ2hvdmVyJyA/IDI1MCA6IDB9XG4gICAgICAgIGhpZGVPbkNsaWNrT3V0c2lkZVxuICAgICAgICBhcmlhLWxhYmVsPXthcmlhTGFiZWx9XG4gICAgICAgIGNsYXNzTmFtZT17Y2xhc3NOYW1lfVxuICAgICAgICB2aXNpYmxlPXt0cmlnZ2VyTWV0aG9kID09PSAnY2xpY2snID8gaXNWaXNpYmxlIDogdW5kZWZpbmVkfVxuICAgICAgICBwbGFjZW1lbnQ9e3BsYWNlbWVudH1cbiAgICAgICAgaGFzQXJyb3c9e2hhc0Fycm93fVxuICAgICAgICBkYXRhLWhhcy1hcnJvdz17aGFzQXJyb3d9XG4gICAgICAgIHJvbGU9XCJkaWFsb2dcIlxuICAgICAgICBpZD17ZmluYWxJZH1cbiAgICAgICAgcmVmPXtwb3B1cFJlZn1cbiAgICAgICAgb25DbG9zZT17KCkgPT4ge1xuICAgICAgICAgIHNldElzVmlzaWJsZShmYWxzZSlcbiAgICAgICAgICBzZXRMb2NhbENoaWxkKG51bGwpXG4gICAgICAgIH19XG4gICAgICAgIHRhYkluZGV4PXstMX1cbiAgICAgICAgbWF4SGVpZ2h0PXttYXhIZWlnaHQgPz8gJzMwcmVtJ31cbiAgICAgICAgbWF4V2lkdGg9e21heFdpZHRofVxuICAgICAgICBzZWFyY2hhYmxlPXtzZWFyY2hhYmxlfVxuICAgICAgICBzaXplPXtzaXplfVxuICAgICAgICB0ZXh0PXtcbiAgICAgICAgICA8ZGl2PlxuICAgICAgICAgICAge3NlYXJjaGFibGUgJiYgdHlwZW9mIGNoaWxkcmVuICE9PSAnZnVuY3Rpb24nID8gKFxuICAgICAgICAgICAgICA8U3R5bGVkU2VhcmNoSW5wdXRcbiAgICAgICAgICAgICAgICBzaXplPVwic21hbGxcIlxuICAgICAgICAgICAgICAgIG9uU2VhcmNoPXtvblNlYXJjaH1cbiAgICAgICAgICAgICAgICByZWY9e3NlYXJjaElucHV0UmVmfVxuICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgKSA6IG51bGx9XG4gICAgICAgICAgICA8TWVudUxpc3RcbiAgICAgICAgICAgICAgZGF0YS10ZXN0aWQ9e2RhdGFUZXN0SWR9XG4gICAgICAgICAgICAgIGNsYXNzTmFtZT17Y2xhc3NOYW1lfVxuICAgICAgICAgICAgICByb2xlPVwibWVudVwiXG4gICAgICAgICAgICA+XG4gICAgICAgICAgICAgIHtmaW5hbENoaWxkfVxuICAgICAgICAgICAgPC9NZW51TGlzdD5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgfVxuICAgICAgICBwb3J0YWxUYXJnZXQ9e3BvcnRhbFRhcmdldH1cbiAgICAgICAgZHluYW1pY0RvbVJlbmRlcmluZz17ZHluYW1pY0RvbVJlbmRlcmluZ31cbiAgICAgICAgYWxpZ249e2FsaWdufVxuICAgICAgPlxuICAgICAgICB7ZmluYWxEaXNjbG9zdXJlfVxuICAgICAgPC9TdHlsZWRQb3B1cD5cbiAgICApXG4gIH0sXG4pXG4iXX0= */"));
33
+ }) => `${theme.space["0.25"]} 0`, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/MenuContent.tsx"],"names":[],"mappings":"AAwBsD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/MenuContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type { ButtonHTMLAttributes, MouseEvent, ReactNode, Ref } from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { SearchInput } from '../SearchInput'\nimport { Stack } from '../Stack'\nimport { useMenu } from './MenuProvider'\nimport { SIZES } from './constants'\nimport { searchChildren } from './helpers'\nimport type { MenuProps } from './types'\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size', 'searchable'].includes(prop),\n})<{ size?: keyof typeof SIZES; searchable: boolean }>`\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) => theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  min-width: ${SIZES.small};\n  max-width: ${SIZES.large};\n  ${({ size }) => (size ? `width: ${SIZES[size]};` : null)}\n  ${({ searchable }) => (searchable ? `min-width: 20rem` : null)};\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n`\n\nconst MenuList = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\nconst StyledSearchInput = styled(SearchInput)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nexport const Menu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget = document.body,\n      size,\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n      searchable = false,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const { isVisible, setIsVisible } = useMenu()\n    const searchInputRef = useRef<HTMLInputElement>(null)\n    const [localChild, setLocalChild] = useState<ReactNode[] | null>(null)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const disclosureRef = useRef<HTMLButtonElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      'aria-haspopup': 'dialog',\n      'aria-expanded': isVisible,\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    const onSearch = useCallback(\n      (value: string) => {\n        if (typeof children === 'object') {\n          setLocalChild(searchChildren(children, value))\n        }\n      },\n      [children],\n    )\n\n    useEffect(() => {\n      if (isVisible && searchable) {\n        setTimeout(() => {\n          searchInputRef.current?.focus()\n        }, 50)\n      }\n    }, [isVisible, searchable])\n\n    const finalChild = useMemo(() => {\n      if (typeof children === 'function') {\n        return children({ toggle: () => setIsVisible(!isVisible) })\n      }\n\n      if (searchable && localChild) {\n        return localChild\n      }\n\n      return children\n    }, [children, isVisible, localChild, searchable, setIsVisible])\n\n    return (\n      <StyledPopup\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        hideOnClickOutside\n        aria-label={ariaLabel}\n        className={className}\n        visible={triggerMethod === 'click' ? isVisible : undefined}\n        placement={placement}\n        hasArrow={hasArrow}\n        data-has-arrow={hasArrow}\n        role=\"dialog\"\n        id={finalId}\n        ref={popupRef}\n        onClose={() => {\n          setIsVisible(false)\n          setLocalChild(null)\n        }}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '30rem'}\n        maxWidth={maxWidth}\n        searchable={searchable}\n        size={size}\n        text={\n          <div>\n            {searchable && typeof children !== 'function' ? (\n              <StyledSearchInput\n                size=\"small\"\n                onSearch={onSearch}\n                ref={searchInputRef}\n              />\n            ) : null}\n            <MenuList\n              data-testid={dataTestId}\n              className={className}\n              role=\"menu\"\n            >\n              {finalChild}\n            </MenuList>\n          </div>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n        align={align}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n"]} */"));
34
34
  const MenuList = /* @__PURE__ */ _styled__default.default(index$1.Stack, process.env.NODE_ENV === "production" ? {
35
35
  target: "e1rdim8w1"
36
36
  } : {
@@ -42,7 +42,7 @@ const MenuList = /* @__PURE__ */ _styled__default.default(index$1.Stack, process
42
42
  theme
43
43
  }) => theme.colors.neutral.text, ";border-radius:", ({
44
44
  theme
45
- }) => theme.radii.default, ";position:relative;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01lbnVWMi9NZW51Q29udGVudC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBMEM4QiIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9NZW51VjIvTWVudUNvbnRlbnQudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7IEJ1dHRvbkhUTUxBdHRyaWJ1dGVzLCBNb3VzZUV2ZW50LCBSZWFjdE5vZGUsIFJlZiB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHtcbiAgY2xvbmVFbGVtZW50LFxuICBmb3J3YXJkUmVmLFxuICBpc1ZhbGlkRWxlbWVudCxcbiAgdXNlQ2FsbGJhY2ssXG4gIHVzZUVmZmVjdCxcbiAgdXNlSWQsXG4gIHVzZUltcGVyYXRpdmVIYW5kbGUsXG4gIHVzZU1lbW8sXG4gIHVzZVJlZixcbiAgdXNlU3RhdGUsXG59IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgUG9wdXAgfSBmcm9tICcuLi9Qb3B1cCdcbmltcG9ydCB7IFNlYXJjaElucHV0IH0gZnJvbSAnLi4vU2VhcmNoSW5wdXQnXG5pbXBvcnQgeyBTdGFjayB9IGZyb20gJy4uL1N0YWNrJ1xuaW1wb3J0IHsgdXNlTWVudSB9IGZyb20gJy4vTWVudVByb3ZpZGVyJ1xuaW1wb3J0IHsgU0laRVMgfSBmcm9tICcuL2NvbnN0YW50cydcbmltcG9ydCB7IHNlYXJjaENoaWxkcmVuIH0gZnJvbSAnLi9oZWxwZXJzJ1xuaW1wb3J0IHR5cGUgeyBNZW51UHJvcHMgfSBmcm9tICcuL3R5cGVzJ1xuXG5jb25zdCBTdHlsZWRQb3B1cCA9IHN0eWxlZChQb3B1cCwge1xuICBzaG91bGRGb3J3YXJkUHJvcDogcHJvcCA9PiAhWydzaXplJywgJ3NlYXJjaGFibGUnXS5pbmNsdWRlcyhwcm9wKSxcbn0pPHsgc2l6ZToga2V5b2YgdHlwZW9mIFNJWkVTOyBzZWFyY2hhYmxlOiBib29sZWFuIH0+YFxuICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLmNvbG9ycy5vdGhlci5lbGV2YXRpb24uYmFja2dyb3VuZC5yYWlzZWR9O1xuICBib3gtc2hhZG93OiAkeyh7IHRoZW1lIH0pID0+IGAke3RoZW1lLnNoYWRvd3MucmFpc2VkWzBdfSwgJHt0aGVtZS5zaGFkb3dzLnJhaXNlZFsxXX1gfTtcbiAgcGFkZGluZzogMDtcblxuICAmW2RhdGEtaGFzLWFycm93PSd0cnVlJ10ge1xuICAgICY6OmFmdGVyIHtcbiAgICAgIGJvcmRlci1jb2xvcjogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5jb2xvcnMub3RoZXIuZWxldmF0aW9uLmJhY2tncm91bmQucmFpc2VkfVxuICAgICAgICB0cmFuc3BhcmVudCB0cmFuc3BhcmVudCB0cmFuc3BhcmVudDtcbiAgICB9XG4gIH1cblxuICB3aWR0aDogJHsoeyBzaXplIH0pID0+IFNJWkVTW3NpemVdfTtcbiAgbWF4LXdpZHRoOiBub25lO1xuICAkeyh7IHNlYXJjaGFibGUgfSkgPT4gKHNlYXJjaGFibGUgPyBgbWluLXdpZHRoOiAyMHJlbWAgOiBudWxsKX07XG4gIHBhZGRpbmc6ICR7KHsgdGhlbWUgfSkgPT4gYCR7dGhlbWUuc3BhY2VbJzAuMjUnXX0gMGB9O1xuYFxuXG5jb25zdCBNZW51TGlzdCA9IHN0eWxlZChTdGFjaylgXG4gIG92ZXJmbG93LXk6IGF1dG87XG4gIG92ZXJmbG93LXg6IGhpZGRlbjtcbiAgJjphZnRlcixcbiAgJjpiZWZvcmUge1xuICAgIGJvcmRlcjogc29saWQgdHJhbnNwYXJlbnQ7XG4gICAgYm9yZGVyLXdpZHRoOiA5cHg7XG4gICAgY29udGVudDogJyAnO1xuICAgIGhlaWdodDogMDtcbiAgICB3aWR0aDogMDtcbiAgICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gICAgcG9pbnRlci1ldmVudHM6IG5vbmU7XG4gIH1cblxuICAmOmFmdGVyIHtcbiAgICBib3JkZXItY29sb3I6IHRyYW5zcGFyZW50O1xuICB9XG4gICY6YmVmb3JlIHtcbiAgICBib3JkZXItY29sb3I6IHRyYW5zcGFyZW50O1xuICB9XG4gIGJhY2tncm91bmQtY29sb3I6ICR7KHsgdGhlbWUgfSkgPT5cbiAgICB0aGVtZS5jb2xvcnMub3RoZXIuZWxldmF0aW9uLmJhY2tncm91bmQucmFpc2VkfTtcbiAgY29sb3I6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuY29sb3JzLm5ldXRyYWwudGV4dH07XG4gIGJvcmRlci1yYWRpdXM6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUucmFkaWkuZGVmYXVsdH07XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbmBcblxuY29uc3QgU3R5bGVkU2VhcmNoSW5wdXQgPSBzdHlsZWQoU2VhcmNoSW5wdXQpYFxuICBwYWRkaW5nOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLnNwYWNlWycxJ119O1xuYFxuXG5leHBvcnQgY29uc3QgTWVudSA9IGZvcndhcmRSZWYoXG4gIChcbiAgICB7XG4gICAgICBpZCxcbiAgICAgIGFyaWFMYWJlbCA9ICdNZW51JyxcbiAgICAgIGNoaWxkcmVuLFxuICAgICAgZGlzY2xvc3VyZSxcbiAgICAgIGhhc0Fycm93ID0gZmFsc2UsXG4gICAgICBwbGFjZW1lbnQgPSAnYm90dG9tJyxcbiAgICAgIGNsYXNzTmFtZSxcbiAgICAgICdkYXRhLXRlc3RpZCc6IGRhdGFUZXN0SWQsXG4gICAgICBtYXhIZWlnaHQsXG4gICAgICBtYXhXaWR0aCxcbiAgICAgIHBvcnRhbFRhcmdldCA9IGRvY3VtZW50LmJvZHksXG4gICAgICBzaXplID0gJ3NtYWxsJyxcbiAgICAgIHRyaWdnZXJNZXRob2QgPSAnY2xpY2snLFxuICAgICAgZHluYW1pY0RvbVJlbmRlcmluZyxcbiAgICAgIGFsaWduLFxuICAgICAgc2VhcmNoYWJsZSA9IGZhbHNlLFxuICAgIH06IE1lbnVQcm9wcyxcbiAgICByZWY6IFJlZjxIVE1MQnV0dG9uRWxlbWVudCB8IG51bGw+LFxuICApID0+IHtcbiAgICBjb25zdCB7IGlzVmlzaWJsZSwgc2V0SXNWaXNpYmxlIH0gPSB1c2VNZW51KClcbiAgICBjb25zdCBzZWFyY2hJbnB1dFJlZiA9IHVzZVJlZjxIVE1MSW5wdXRFbGVtZW50PihudWxsKVxuICAgIGNvbnN0IFtsb2NhbENoaWxkLCBzZXRMb2NhbENoaWxkXSA9IHVzZVN0YXRlPFJlYWN0Tm9kZVtdIHwgbnVsbD4obnVsbClcbiAgICBjb25zdCBwb3B1cFJlZiA9IHVzZVJlZjxIVE1MRGl2RWxlbWVudD4obnVsbClcbiAgICBjb25zdCBkaXNjbG9zdXJlUmVmID0gdXNlUmVmPEhUTUxCdXR0b25FbGVtZW50PihudWxsKVxuICAgIGNvbnN0IHRlbXBJZCA9IHVzZUlkKClcbiAgICBjb25zdCBmaW5hbElkID0gYG1lbnUtJHtpZCA/PyB0ZW1wSWR9YFxuXG4gICAgLy8gaWYgeW91IG5lZWQgZGlhbG9nIGluc2lkZSB5b3VyIGNvbXBvbmVudCwgdXNlIGZ1bmN0aW9uLCBvdGhlcndpc2UgY29tcG9uZW50IGlzIGZpbmVcbiAgICBjb25zdCB0YXJnZXQgPSBpc1ZhbGlkRWxlbWVudDxCdXR0b25IVE1MQXR0cmlidXRlczxIVE1MQnV0dG9uRWxlbWVudD4+KFxuICAgICAgZGlzY2xvc3VyZSxcbiAgICApXG4gICAgICA/IGRpc2Nsb3N1cmVcbiAgICAgIDogZGlzY2xvc3VyZSh7IHZpc2libGU6IGlzVmlzaWJsZSB9KVxuICAgIGNvbnN0IGlubmVyUmVmID0gdXNlUmVmKHRhcmdldCBhcyB1bmtub3duIGFzIEhUTUxCdXR0b25FbGVtZW50KVxuICAgIHVzZUltcGVyYXRpdmVIYW5kbGUocmVmLCAoKSA9PiBpbm5lclJlZi5jdXJyZW50KVxuXG4gICAgY29uc3QgZmluYWxEaXNjbG9zdXJlID0gY2xvbmVFbGVtZW50KHRhcmdldCwge1xuICAgICAgb25DbGljazogKGV2ZW50OiBNb3VzZUV2ZW50PEhUTUxCdXR0b25FbGVtZW50PikgPT4ge1xuICAgICAgICB0YXJnZXQucHJvcHMub25DbGljaz8uKGV2ZW50KVxuICAgICAgICBzZXRJc1Zpc2libGUoIWlzVmlzaWJsZSlcbiAgICAgIH0sXG4gICAgICAnYXJpYS1oYXNwb3B1cCc6ICdkaWFsb2cnLFxuICAgICAgJ2FyaWEtZXhwYW5kZWQnOiBpc1Zpc2libGUsXG4gICAgICAvLyBAdHMtZXhwZWN0LWVycm9yIG5vdCBzdXJlIGhvdyB0byBmaXggdGhpc1xuICAgICAgcmVmOiBkaXNjbG9zdXJlUmVmLFxuICAgIH0pXG5cbiAgICBjb25zdCBvblNlYXJjaCA9IHVzZUNhbGxiYWNrKFxuICAgICAgKHZhbHVlOiBzdHJpbmcpID0+IHtcbiAgICAgICAgaWYgKHR5cGVvZiBjaGlsZHJlbiA9PT0gJ29iamVjdCcpIHtcbiAgICAgICAgICBzZXRMb2NhbENoaWxkKHNlYXJjaENoaWxkcmVuKGNoaWxkcmVuLCB2YWx1ZSkpXG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgICBbY2hpbGRyZW5dLFxuICAgIClcblxuICAgIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgICBpZiAoaXNWaXNpYmxlICYmIHNlYXJjaGFibGUpIHtcbiAgICAgICAgc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICAgICAgc2VhcmNoSW5wdXRSZWYuY3VycmVudD8uZm9jdXMoKVxuICAgICAgICB9LCA1MClcbiAgICAgIH1cbiAgICB9LCBbaXNWaXNpYmxlLCBzZWFyY2hhYmxlXSlcblxuICAgIGNvbnN0IGZpbmFsQ2hpbGQgPSB1c2VNZW1vKCgpID0+IHtcbiAgICAgIGlmICh0eXBlb2YgY2hpbGRyZW4gPT09ICdmdW5jdGlvbicpIHtcbiAgICAgICAgcmV0dXJuIGNoaWxkcmVuKHsgdG9nZ2xlOiAoKSA9PiBzZXRJc1Zpc2libGUoIWlzVmlzaWJsZSkgfSlcbiAgICAgIH1cblxuICAgICAgaWYgKHNlYXJjaGFibGUgJiYgbG9jYWxDaGlsZCkge1xuICAgICAgICByZXR1cm4gbG9jYWxDaGlsZFxuICAgICAgfVxuXG4gICAgICByZXR1cm4gY2hpbGRyZW5cbiAgICB9LCBbY2hpbGRyZW4sIGlzVmlzaWJsZSwgbG9jYWxDaGlsZCwgc2VhcmNoYWJsZSwgc2V0SXNWaXNpYmxlXSlcblxuICAgIHJldHVybiAoXG4gICAgICA8U3R5bGVkUG9wdXBcbiAgICAgICAgZGVib3VuY2VEZWxheT17dHJpZ2dlck1ldGhvZCA9PT0gJ2hvdmVyJyA/IDI1MCA6IDB9XG4gICAgICAgIGhpZGVPbkNsaWNrT3V0c2lkZVxuICAgICAgICBhcmlhLWxhYmVsPXthcmlhTGFiZWx9XG4gICAgICAgIGNsYXNzTmFtZT17Y2xhc3NOYW1lfVxuICAgICAgICB2aXNpYmxlPXt0cmlnZ2VyTWV0aG9kID09PSAnY2xpY2snID8gaXNWaXNpYmxlIDogdW5kZWZpbmVkfVxuICAgICAgICBwbGFjZW1lbnQ9e3BsYWNlbWVudH1cbiAgICAgICAgaGFzQXJyb3c9e2hhc0Fycm93fVxuICAgICAgICBkYXRhLWhhcy1hcnJvdz17aGFzQXJyb3d9XG4gICAgICAgIHJvbGU9XCJkaWFsb2dcIlxuICAgICAgICBpZD17ZmluYWxJZH1cbiAgICAgICAgcmVmPXtwb3B1cFJlZn1cbiAgICAgICAgb25DbG9zZT17KCkgPT4ge1xuICAgICAgICAgIHNldElzVmlzaWJsZShmYWxzZSlcbiAgICAgICAgICBzZXRMb2NhbENoaWxkKG51bGwpXG4gICAgICAgIH19XG4gICAgICAgIHRhYkluZGV4PXstMX1cbiAgICAgICAgbWF4SGVpZ2h0PXttYXhIZWlnaHQgPz8gJzMwcmVtJ31cbiAgICAgICAgbWF4V2lkdGg9e21heFdpZHRofVxuICAgICAgICBzZWFyY2hhYmxlPXtzZWFyY2hhYmxlfVxuICAgICAgICBzaXplPXtzaXplfVxuICAgICAgICB0ZXh0PXtcbiAgICAgICAgICA8ZGl2PlxuICAgICAgICAgICAge3NlYXJjaGFibGUgJiYgdHlwZW9mIGNoaWxkcmVuICE9PSAnZnVuY3Rpb24nID8gKFxuICAgICAgICAgICAgICA8U3R5bGVkU2VhcmNoSW5wdXRcbiAgICAgICAgICAgICAgICBzaXplPVwic21hbGxcIlxuICAgICAgICAgICAgICAgIG9uU2VhcmNoPXtvblNlYXJjaH1cbiAgICAgICAgICAgICAgICByZWY9e3NlYXJjaElucHV0UmVmfVxuICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgKSA6IG51bGx9XG4gICAgICAgICAgICA8TWVudUxpc3RcbiAgICAgICAgICAgICAgZGF0YS10ZXN0aWQ9e2RhdGFUZXN0SWR9XG4gICAgICAgICAgICAgIGNsYXNzTmFtZT17Y2xhc3NOYW1lfVxuICAgICAgICAgICAgICByb2xlPVwibWVudVwiXG4gICAgICAgICAgICA+XG4gICAgICAgICAgICAgIHtmaW5hbENoaWxkfVxuICAgICAgICAgICAgPC9NZW51TGlzdD5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgfVxuICAgICAgICBwb3J0YWxUYXJnZXQ9e3BvcnRhbFRhcmdldH1cbiAgICAgICAgZHluYW1pY0RvbVJlbmRlcmluZz17ZHluYW1pY0RvbVJlbmRlcmluZ31cbiAgICAgICAgYWxpZ249e2FsaWdufVxuICAgICAgPlxuICAgICAgICB7ZmluYWxEaXNjbG9zdXJlfVxuICAgICAgPC9TdHlsZWRQb3B1cD5cbiAgICApXG4gIH0sXG4pXG4iXX0= */"));
45
+ }) => theme.radii.default, ";position:relative;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/MenuContent.tsx"],"names":[],"mappings":"AA2C8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/MenuContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type { ButtonHTMLAttributes, MouseEvent, ReactNode, Ref } from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { SearchInput } from '../SearchInput'\nimport { Stack } from '../Stack'\nimport { useMenu } from './MenuProvider'\nimport { SIZES } from './constants'\nimport { searchChildren } from './helpers'\nimport type { MenuProps } from './types'\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size', 'searchable'].includes(prop),\n})<{ size?: keyof typeof SIZES; searchable: boolean }>`\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) => theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  min-width: ${SIZES.small};\n  max-width: ${SIZES.large};\n  ${({ size }) => (size ? `width: ${SIZES[size]};` : null)}\n  ${({ searchable }) => (searchable ? `min-width: 20rem` : null)};\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n`\n\nconst MenuList = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\nconst StyledSearchInput = styled(SearchInput)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nexport const Menu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget = document.body,\n      size,\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n      searchable = false,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const { isVisible, setIsVisible } = useMenu()\n    const searchInputRef = useRef<HTMLInputElement>(null)\n    const [localChild, setLocalChild] = useState<ReactNode[] | null>(null)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const disclosureRef = useRef<HTMLButtonElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      'aria-haspopup': 'dialog',\n      'aria-expanded': isVisible,\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    const onSearch = useCallback(\n      (value: string) => {\n        if (typeof children === 'object') {\n          setLocalChild(searchChildren(children, value))\n        }\n      },\n      [children],\n    )\n\n    useEffect(() => {\n      if (isVisible && searchable) {\n        setTimeout(() => {\n          searchInputRef.current?.focus()\n        }, 50)\n      }\n    }, [isVisible, searchable])\n\n    const finalChild = useMemo(() => {\n      if (typeof children === 'function') {\n        return children({ toggle: () => setIsVisible(!isVisible) })\n      }\n\n      if (searchable && localChild) {\n        return localChild\n      }\n\n      return children\n    }, [children, isVisible, localChild, searchable, setIsVisible])\n\n    return (\n      <StyledPopup\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        hideOnClickOutside\n        aria-label={ariaLabel}\n        className={className}\n        visible={triggerMethod === 'click' ? isVisible : undefined}\n        placement={placement}\n        hasArrow={hasArrow}\n        data-has-arrow={hasArrow}\n        role=\"dialog\"\n        id={finalId}\n        ref={popupRef}\n        onClose={() => {\n          setIsVisible(false)\n          setLocalChild(null)\n        }}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '30rem'}\n        maxWidth={maxWidth}\n        searchable={searchable}\n        size={size}\n        text={\n          <div>\n            {searchable && typeof children !== 'function' ? (\n              <StyledSearchInput\n                size=\"small\"\n                onSearch={onSearch}\n                ref={searchInputRef}\n              />\n            ) : null}\n            <MenuList\n              data-testid={dataTestId}\n              className={className}\n              role=\"menu\"\n            >\n              {finalChild}\n            </MenuList>\n          </div>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n        align={align}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n"]} */"));
46
46
  const StyledSearchInput = /* @__PURE__ */ _styled__default.default(index$2.SearchInput, process.env.NODE_ENV === "production" ? {
47
47
  target: "e1rdim8w0"
48
48
  } : {
@@ -50,7 +50,7 @@ const StyledSearchInput = /* @__PURE__ */ _styled__default.default(index$2.Searc
50
50
  label: "StyledSearchInput"
51
51
  })("padding:", ({
52
52
  theme
53
- }) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01lbnVWMi9NZW51Q29udGVudC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBcUU2QyIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9NZW51VjIvTWVudUNvbnRlbnQudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7IEJ1dHRvbkhUTUxBdHRyaWJ1dGVzLCBNb3VzZUV2ZW50LCBSZWFjdE5vZGUsIFJlZiB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHtcbiAgY2xvbmVFbGVtZW50LFxuICBmb3J3YXJkUmVmLFxuICBpc1ZhbGlkRWxlbWVudCxcbiAgdXNlQ2FsbGJhY2ssXG4gIHVzZUVmZmVjdCxcbiAgdXNlSWQsXG4gIHVzZUltcGVyYXRpdmVIYW5kbGUsXG4gIHVzZU1lbW8sXG4gIHVzZVJlZixcbiAgdXNlU3RhdGUsXG59IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgUG9wdXAgfSBmcm9tICcuLi9Qb3B1cCdcbmltcG9ydCB7IFNlYXJjaElucHV0IH0gZnJvbSAnLi4vU2VhcmNoSW5wdXQnXG5pbXBvcnQgeyBTdGFjayB9IGZyb20gJy4uL1N0YWNrJ1xuaW1wb3J0IHsgdXNlTWVudSB9IGZyb20gJy4vTWVudVByb3ZpZGVyJ1xuaW1wb3J0IHsgU0laRVMgfSBmcm9tICcuL2NvbnN0YW50cydcbmltcG9ydCB7IHNlYXJjaENoaWxkcmVuIH0gZnJvbSAnLi9oZWxwZXJzJ1xuaW1wb3J0IHR5cGUgeyBNZW51UHJvcHMgfSBmcm9tICcuL3R5cGVzJ1xuXG5jb25zdCBTdHlsZWRQb3B1cCA9IHN0eWxlZChQb3B1cCwge1xuICBzaG91bGRGb3J3YXJkUHJvcDogcHJvcCA9PiAhWydzaXplJywgJ3NlYXJjaGFibGUnXS5pbmNsdWRlcyhwcm9wKSxcbn0pPHsgc2l6ZToga2V5b2YgdHlwZW9mIFNJWkVTOyBzZWFyY2hhYmxlOiBib29sZWFuIH0+YFxuICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLmNvbG9ycy5vdGhlci5lbGV2YXRpb24uYmFja2dyb3VuZC5yYWlzZWR9O1xuICBib3gtc2hhZG93OiAkeyh7IHRoZW1lIH0pID0+IGAke3RoZW1lLnNoYWRvd3MucmFpc2VkWzBdfSwgJHt0aGVtZS5zaGFkb3dzLnJhaXNlZFsxXX1gfTtcbiAgcGFkZGluZzogMDtcblxuICAmW2RhdGEtaGFzLWFycm93PSd0cnVlJ10ge1xuICAgICY6OmFmdGVyIHtcbiAgICAgIGJvcmRlci1jb2xvcjogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5jb2xvcnMub3RoZXIuZWxldmF0aW9uLmJhY2tncm91bmQucmFpc2VkfVxuICAgICAgICB0cmFuc3BhcmVudCB0cmFuc3BhcmVudCB0cmFuc3BhcmVudDtcbiAgICB9XG4gIH1cblxuICB3aWR0aDogJHsoeyBzaXplIH0pID0+IFNJWkVTW3NpemVdfTtcbiAgbWF4LXdpZHRoOiBub25lO1xuICAkeyh7IHNlYXJjaGFibGUgfSkgPT4gKHNlYXJjaGFibGUgPyBgbWluLXdpZHRoOiAyMHJlbWAgOiBudWxsKX07XG4gIHBhZGRpbmc6ICR7KHsgdGhlbWUgfSkgPT4gYCR7dGhlbWUuc3BhY2VbJzAuMjUnXX0gMGB9O1xuYFxuXG5jb25zdCBNZW51TGlzdCA9IHN0eWxlZChTdGFjaylgXG4gIG92ZXJmbG93LXk6IGF1dG87XG4gIG92ZXJmbG93LXg6IGhpZGRlbjtcbiAgJjphZnRlcixcbiAgJjpiZWZvcmUge1xuICAgIGJvcmRlcjogc29saWQgdHJhbnNwYXJlbnQ7XG4gICAgYm9yZGVyLXdpZHRoOiA5cHg7XG4gICAgY29udGVudDogJyAnO1xuICAgIGhlaWdodDogMDtcbiAgICB3aWR0aDogMDtcbiAgICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gICAgcG9pbnRlci1ldmVudHM6IG5vbmU7XG4gIH1cblxuICAmOmFmdGVyIHtcbiAgICBib3JkZXItY29sb3I6IHRyYW5zcGFyZW50O1xuICB9XG4gICY6YmVmb3JlIHtcbiAgICBib3JkZXItY29sb3I6IHRyYW5zcGFyZW50O1xuICB9XG4gIGJhY2tncm91bmQtY29sb3I6ICR7KHsgdGhlbWUgfSkgPT5cbiAgICB0aGVtZS5jb2xvcnMub3RoZXIuZWxldmF0aW9uLmJhY2tncm91bmQucmFpc2VkfTtcbiAgY29sb3I6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuY29sb3JzLm5ldXRyYWwudGV4dH07XG4gIGJvcmRlci1yYWRpdXM6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUucmFkaWkuZGVmYXVsdH07XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbmBcblxuY29uc3QgU3R5bGVkU2VhcmNoSW5wdXQgPSBzdHlsZWQoU2VhcmNoSW5wdXQpYFxuICBwYWRkaW5nOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLnNwYWNlWycxJ119O1xuYFxuXG5leHBvcnQgY29uc3QgTWVudSA9IGZvcndhcmRSZWYoXG4gIChcbiAgICB7XG4gICAgICBpZCxcbiAgICAgIGFyaWFMYWJlbCA9ICdNZW51JyxcbiAgICAgIGNoaWxkcmVuLFxuICAgICAgZGlzY2xvc3VyZSxcbiAgICAgIGhhc0Fycm93ID0gZmFsc2UsXG4gICAgICBwbGFjZW1lbnQgPSAnYm90dG9tJyxcbiAgICAgIGNsYXNzTmFtZSxcbiAgICAgICdkYXRhLXRlc3RpZCc6IGRhdGFUZXN0SWQsXG4gICAgICBtYXhIZWlnaHQsXG4gICAgICBtYXhXaWR0aCxcbiAgICAgIHBvcnRhbFRhcmdldCA9IGRvY3VtZW50LmJvZHksXG4gICAgICBzaXplID0gJ3NtYWxsJyxcbiAgICAgIHRyaWdnZXJNZXRob2QgPSAnY2xpY2snLFxuICAgICAgZHluYW1pY0RvbVJlbmRlcmluZyxcbiAgICAgIGFsaWduLFxuICAgICAgc2VhcmNoYWJsZSA9IGZhbHNlLFxuICAgIH06IE1lbnVQcm9wcyxcbiAgICByZWY6IFJlZjxIVE1MQnV0dG9uRWxlbWVudCB8IG51bGw+LFxuICApID0+IHtcbiAgICBjb25zdCB7IGlzVmlzaWJsZSwgc2V0SXNWaXNpYmxlIH0gPSB1c2VNZW51KClcbiAgICBjb25zdCBzZWFyY2hJbnB1dFJlZiA9IHVzZVJlZjxIVE1MSW5wdXRFbGVtZW50PihudWxsKVxuICAgIGNvbnN0IFtsb2NhbENoaWxkLCBzZXRMb2NhbENoaWxkXSA9IHVzZVN0YXRlPFJlYWN0Tm9kZVtdIHwgbnVsbD4obnVsbClcbiAgICBjb25zdCBwb3B1cFJlZiA9IHVzZVJlZjxIVE1MRGl2RWxlbWVudD4obnVsbClcbiAgICBjb25zdCBkaXNjbG9zdXJlUmVmID0gdXNlUmVmPEhUTUxCdXR0b25FbGVtZW50PihudWxsKVxuICAgIGNvbnN0IHRlbXBJZCA9IHVzZUlkKClcbiAgICBjb25zdCBmaW5hbElkID0gYG1lbnUtJHtpZCA/PyB0ZW1wSWR9YFxuXG4gICAgLy8gaWYgeW91IG5lZWQgZGlhbG9nIGluc2lkZSB5b3VyIGNvbXBvbmVudCwgdXNlIGZ1bmN0aW9uLCBvdGhlcndpc2UgY29tcG9uZW50IGlzIGZpbmVcbiAgICBjb25zdCB0YXJnZXQgPSBpc1ZhbGlkRWxlbWVudDxCdXR0b25IVE1MQXR0cmlidXRlczxIVE1MQnV0dG9uRWxlbWVudD4+KFxuICAgICAgZGlzY2xvc3VyZSxcbiAgICApXG4gICAgICA/IGRpc2Nsb3N1cmVcbiAgICAgIDogZGlzY2xvc3VyZSh7IHZpc2libGU6IGlzVmlzaWJsZSB9KVxuICAgIGNvbnN0IGlubmVyUmVmID0gdXNlUmVmKHRhcmdldCBhcyB1bmtub3duIGFzIEhUTUxCdXR0b25FbGVtZW50KVxuICAgIHVzZUltcGVyYXRpdmVIYW5kbGUocmVmLCAoKSA9PiBpbm5lclJlZi5jdXJyZW50KVxuXG4gICAgY29uc3QgZmluYWxEaXNjbG9zdXJlID0gY2xvbmVFbGVtZW50KHRhcmdldCwge1xuICAgICAgb25DbGljazogKGV2ZW50OiBNb3VzZUV2ZW50PEhUTUxCdXR0b25FbGVtZW50PikgPT4ge1xuICAgICAgICB0YXJnZXQucHJvcHMub25DbGljaz8uKGV2ZW50KVxuICAgICAgICBzZXRJc1Zpc2libGUoIWlzVmlzaWJsZSlcbiAgICAgIH0sXG4gICAgICAnYXJpYS1oYXNwb3B1cCc6ICdkaWFsb2cnLFxuICAgICAgJ2FyaWEtZXhwYW5kZWQnOiBpc1Zpc2libGUsXG4gICAgICAvLyBAdHMtZXhwZWN0LWVycm9yIG5vdCBzdXJlIGhvdyB0byBmaXggdGhpc1xuICAgICAgcmVmOiBkaXNjbG9zdXJlUmVmLFxuICAgIH0pXG5cbiAgICBjb25zdCBvblNlYXJjaCA9IHVzZUNhbGxiYWNrKFxuICAgICAgKHZhbHVlOiBzdHJpbmcpID0+IHtcbiAgICAgICAgaWYgKHR5cGVvZiBjaGlsZHJlbiA9PT0gJ29iamVjdCcpIHtcbiAgICAgICAgICBzZXRMb2NhbENoaWxkKHNlYXJjaENoaWxkcmVuKGNoaWxkcmVuLCB2YWx1ZSkpXG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgICBbY2hpbGRyZW5dLFxuICAgIClcblxuICAgIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgICBpZiAoaXNWaXNpYmxlICYmIHNlYXJjaGFibGUpIHtcbiAgICAgICAgc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICAgICAgc2VhcmNoSW5wdXRSZWYuY3VycmVudD8uZm9jdXMoKVxuICAgICAgICB9LCA1MClcbiAgICAgIH1cbiAgICB9LCBbaXNWaXNpYmxlLCBzZWFyY2hhYmxlXSlcblxuICAgIGNvbnN0IGZpbmFsQ2hpbGQgPSB1c2VNZW1vKCgpID0+IHtcbiAgICAgIGlmICh0eXBlb2YgY2hpbGRyZW4gPT09ICdmdW5jdGlvbicpIHtcbiAgICAgICAgcmV0dXJuIGNoaWxkcmVuKHsgdG9nZ2xlOiAoKSA9PiBzZXRJc1Zpc2libGUoIWlzVmlzaWJsZSkgfSlcbiAgICAgIH1cblxuICAgICAgaWYgKHNlYXJjaGFibGUgJiYgbG9jYWxDaGlsZCkge1xuICAgICAgICByZXR1cm4gbG9jYWxDaGlsZFxuICAgICAgfVxuXG4gICAgICByZXR1cm4gY2hpbGRyZW5cbiAgICB9LCBbY2hpbGRyZW4sIGlzVmlzaWJsZSwgbG9jYWxDaGlsZCwgc2VhcmNoYWJsZSwgc2V0SXNWaXNpYmxlXSlcblxuICAgIHJldHVybiAoXG4gICAgICA8U3R5bGVkUG9wdXBcbiAgICAgICAgZGVib3VuY2VEZWxheT17dHJpZ2dlck1ldGhvZCA9PT0gJ2hvdmVyJyA/IDI1MCA6IDB9XG4gICAgICAgIGhpZGVPbkNsaWNrT3V0c2lkZVxuICAgICAgICBhcmlhLWxhYmVsPXthcmlhTGFiZWx9XG4gICAgICAgIGNsYXNzTmFtZT17Y2xhc3NOYW1lfVxuICAgICAgICB2aXNpYmxlPXt0cmlnZ2VyTWV0aG9kID09PSAnY2xpY2snID8gaXNWaXNpYmxlIDogdW5kZWZpbmVkfVxuICAgICAgICBwbGFjZW1lbnQ9e3BsYWNlbWVudH1cbiAgICAgICAgaGFzQXJyb3c9e2hhc0Fycm93fVxuICAgICAgICBkYXRhLWhhcy1hcnJvdz17aGFzQXJyb3d9XG4gICAgICAgIHJvbGU9XCJkaWFsb2dcIlxuICAgICAgICBpZD17ZmluYWxJZH1cbiAgICAgICAgcmVmPXtwb3B1cFJlZn1cbiAgICAgICAgb25DbG9zZT17KCkgPT4ge1xuICAgICAgICAgIHNldElzVmlzaWJsZShmYWxzZSlcbiAgICAgICAgICBzZXRMb2NhbENoaWxkKG51bGwpXG4gICAgICAgIH19XG4gICAgICAgIHRhYkluZGV4PXstMX1cbiAgICAgICAgbWF4SGVpZ2h0PXttYXhIZWlnaHQgPz8gJzMwcmVtJ31cbiAgICAgICAgbWF4V2lkdGg9e21heFdpZHRofVxuICAgICAgICBzZWFyY2hhYmxlPXtzZWFyY2hhYmxlfVxuICAgICAgICBzaXplPXtzaXplfVxuICAgICAgICB0ZXh0PXtcbiAgICAgICAgICA8ZGl2PlxuICAgICAgICAgICAge3NlYXJjaGFibGUgJiYgdHlwZW9mIGNoaWxkcmVuICE9PSAnZnVuY3Rpb24nID8gKFxuICAgICAgICAgICAgICA8U3R5bGVkU2VhcmNoSW5wdXRcbiAgICAgICAgICAgICAgICBzaXplPVwic21hbGxcIlxuICAgICAgICAgICAgICAgIG9uU2VhcmNoPXtvblNlYXJjaH1cbiAgICAgICAgICAgICAgICByZWY9e3NlYXJjaElucHV0UmVmfVxuICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgKSA6IG51bGx9XG4gICAgICAgICAgICA8TWVudUxpc3RcbiAgICAgICAgICAgICAgZGF0YS10ZXN0aWQ9e2RhdGFUZXN0SWR9XG4gICAgICAgICAgICAgIGNsYXNzTmFtZT17Y2xhc3NOYW1lfVxuICAgICAgICAgICAgICByb2xlPVwibWVudVwiXG4gICAgICAgICAgICA+XG4gICAgICAgICAgICAgIHtmaW5hbENoaWxkfVxuICAgICAgICAgICAgPC9NZW51TGlzdD5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgfVxuICAgICAgICBwb3J0YWxUYXJnZXQ9e3BvcnRhbFRhcmdldH1cbiAgICAgICAgZHluYW1pY0RvbVJlbmRlcmluZz17ZHluYW1pY0RvbVJlbmRlcmluZ31cbiAgICAgICAgYWxpZ249e2FsaWdufVxuICAgICAgPlxuICAgICAgICB7ZmluYWxEaXNjbG9zdXJlfVxuICAgICAgPC9TdHlsZWRQb3B1cD5cbiAgICApXG4gIH0sXG4pXG4iXX0= */"));
53
+ }) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/MenuContent.tsx"],"names":[],"mappings":"AAsE6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/MenuContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type { ButtonHTMLAttributes, MouseEvent, ReactNode, Ref } from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { SearchInput } from '../SearchInput'\nimport { Stack } from '../Stack'\nimport { useMenu } from './MenuProvider'\nimport { SIZES } from './constants'\nimport { searchChildren } from './helpers'\nimport type { MenuProps } from './types'\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size', 'searchable'].includes(prop),\n})<{ size?: keyof typeof SIZES; searchable: boolean }>`\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) => theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  min-width: ${SIZES.small};\n  max-width: ${SIZES.large};\n  ${({ size }) => (size ? `width: ${SIZES[size]};` : null)}\n  ${({ searchable }) => (searchable ? `min-width: 20rem` : null)};\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n`\n\nconst MenuList = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\nconst StyledSearchInput = styled(SearchInput)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nexport const Menu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget = document.body,\n      size,\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n      searchable = false,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const { isVisible, setIsVisible } = useMenu()\n    const searchInputRef = useRef<HTMLInputElement>(null)\n    const [localChild, setLocalChild] = useState<ReactNode[] | null>(null)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const disclosureRef = useRef<HTMLButtonElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      'aria-haspopup': 'dialog',\n      'aria-expanded': isVisible,\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    const onSearch = useCallback(\n      (value: string) => {\n        if (typeof children === 'object') {\n          setLocalChild(searchChildren(children, value))\n        }\n      },\n      [children],\n    )\n\n    useEffect(() => {\n      if (isVisible && searchable) {\n        setTimeout(() => {\n          searchInputRef.current?.focus()\n        }, 50)\n      }\n    }, [isVisible, searchable])\n\n    const finalChild = useMemo(() => {\n      if (typeof children === 'function') {\n        return children({ toggle: () => setIsVisible(!isVisible) })\n      }\n\n      if (searchable && localChild) {\n        return localChild\n      }\n\n      return children\n    }, [children, isVisible, localChild, searchable, setIsVisible])\n\n    return (\n      <StyledPopup\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        hideOnClickOutside\n        aria-label={ariaLabel}\n        className={className}\n        visible={triggerMethod === 'click' ? isVisible : undefined}\n        placement={placement}\n        hasArrow={hasArrow}\n        data-has-arrow={hasArrow}\n        role=\"dialog\"\n        id={finalId}\n        ref={popupRef}\n        onClose={() => {\n          setIsVisible(false)\n          setLocalChild(null)\n        }}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '30rem'}\n        maxWidth={maxWidth}\n        searchable={searchable}\n        size={size}\n        text={\n          <div>\n            {searchable && typeof children !== 'function' ? (\n              <StyledSearchInput\n                size=\"small\"\n                onSearch={onSearch}\n                ref={searchInputRef}\n              />\n            ) : null}\n            <MenuList\n              data-testid={dataTestId}\n              className={className}\n              role=\"menu\"\n            >\n              {finalChild}\n            </MenuList>\n          </div>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n        align={align}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n"]} */"));
54
54
  const Menu = React.forwardRef(({
55
55
  id,
56
56
  ariaLabel = "Menu",
@@ -63,7 +63,7 @@ const Menu = React.forwardRef(({
63
63
  maxHeight,
64
64
  maxWidth,
65
65
  portalTarget = document.body,
66
- size = "small",
66
+ size,
67
67
  triggerMethod = "click",
68
68
  dynamicDomRendering,
69
69
  align,
@@ -20,13 +20,13 @@ const StyledPopup = /* @__PURE__ */ _styled(Popup, process.env.NODE_ENV === "pro
20
20
  theme
21
21
  }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`, ";padding:0;&[data-has-arrow='true']{&::after{border-color:", ({
22
22
  theme
23
- }) => theme.colors.other.elevation.background.raised, " transparent transparent transparent;}}width:", ({
23
+ }) => theme.colors.other.elevation.background.raised, " transparent transparent transparent;}}min-width:", SIZES.small, ";max-width:", SIZES.large, ";", ({
24
24
  size
25
- }) => SIZES[size], ";max-width:none;", ({
25
+ }) => size ? `width: ${SIZES[size]};` : null, " ", ({
26
26
  searchable
27
27
  }) => searchable ? `min-width: 20rem` : null, ";padding:", ({
28
28
  theme
29
- }) => `${theme.space["0.25"]} 0`, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01lbnVWMi9NZW51Q29udGVudC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBd0JxRCIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9NZW51VjIvTWVudUNvbnRlbnQudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7IEJ1dHRvbkhUTUxBdHRyaWJ1dGVzLCBNb3VzZUV2ZW50LCBSZWFjdE5vZGUsIFJlZiB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHtcbiAgY2xvbmVFbGVtZW50LFxuICBmb3J3YXJkUmVmLFxuICBpc1ZhbGlkRWxlbWVudCxcbiAgdXNlQ2FsbGJhY2ssXG4gIHVzZUVmZmVjdCxcbiAgdXNlSWQsXG4gIHVzZUltcGVyYXRpdmVIYW5kbGUsXG4gIHVzZU1lbW8sXG4gIHVzZVJlZixcbiAgdXNlU3RhdGUsXG59IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgUG9wdXAgfSBmcm9tICcuLi9Qb3B1cCdcbmltcG9ydCB7IFNlYXJjaElucHV0IH0gZnJvbSAnLi4vU2VhcmNoSW5wdXQnXG5pbXBvcnQgeyBTdGFjayB9IGZyb20gJy4uL1N0YWNrJ1xuaW1wb3J0IHsgdXNlTWVudSB9IGZyb20gJy4vTWVudVByb3ZpZGVyJ1xuaW1wb3J0IHsgU0laRVMgfSBmcm9tICcuL2NvbnN0YW50cydcbmltcG9ydCB7IHNlYXJjaENoaWxkcmVuIH0gZnJvbSAnLi9oZWxwZXJzJ1xuaW1wb3J0IHR5cGUgeyBNZW51UHJvcHMgfSBmcm9tICcuL3R5cGVzJ1xuXG5jb25zdCBTdHlsZWRQb3B1cCA9IHN0eWxlZChQb3B1cCwge1xuICBzaG91bGRGb3J3YXJkUHJvcDogcHJvcCA9PiAhWydzaXplJywgJ3NlYXJjaGFibGUnXS5pbmNsdWRlcyhwcm9wKSxcbn0pPHsgc2l6ZToga2V5b2YgdHlwZW9mIFNJWkVTOyBzZWFyY2hhYmxlOiBib29sZWFuIH0+YFxuICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLmNvbG9ycy5vdGhlci5lbGV2YXRpb24uYmFja2dyb3VuZC5yYWlzZWR9O1xuICBib3gtc2hhZG93OiAkeyh7IHRoZW1lIH0pID0+IGAke3RoZW1lLnNoYWRvd3MucmFpc2VkWzBdfSwgJHt0aGVtZS5zaGFkb3dzLnJhaXNlZFsxXX1gfTtcbiAgcGFkZGluZzogMDtcblxuICAmW2RhdGEtaGFzLWFycm93PSd0cnVlJ10ge1xuICAgICY6OmFmdGVyIHtcbiAgICAgIGJvcmRlci1jb2xvcjogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5jb2xvcnMub3RoZXIuZWxldmF0aW9uLmJhY2tncm91bmQucmFpc2VkfVxuICAgICAgICB0cmFuc3BhcmVudCB0cmFuc3BhcmVudCB0cmFuc3BhcmVudDtcbiAgICB9XG4gIH1cblxuICB3aWR0aDogJHsoeyBzaXplIH0pID0+IFNJWkVTW3NpemVdfTtcbiAgbWF4LXdpZHRoOiBub25lO1xuICAkeyh7IHNlYXJjaGFibGUgfSkgPT4gKHNlYXJjaGFibGUgPyBgbWluLXdpZHRoOiAyMHJlbWAgOiBudWxsKX07XG4gIHBhZGRpbmc6ICR7KHsgdGhlbWUgfSkgPT4gYCR7dGhlbWUuc3BhY2VbJzAuMjUnXX0gMGB9O1xuYFxuXG5jb25zdCBNZW51TGlzdCA9IHN0eWxlZChTdGFjaylgXG4gIG92ZXJmbG93LXk6IGF1dG87XG4gIG92ZXJmbG93LXg6IGhpZGRlbjtcbiAgJjphZnRlcixcbiAgJjpiZWZvcmUge1xuICAgIGJvcmRlcjogc29saWQgdHJhbnNwYXJlbnQ7XG4gICAgYm9yZGVyLXdpZHRoOiA5cHg7XG4gICAgY29udGVudDogJyAnO1xuICAgIGhlaWdodDogMDtcbiAgICB3aWR0aDogMDtcbiAgICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gICAgcG9pbnRlci1ldmVudHM6IG5vbmU7XG4gIH1cblxuICAmOmFmdGVyIHtcbiAgICBib3JkZXItY29sb3I6IHRyYW5zcGFyZW50O1xuICB9XG4gICY6YmVmb3JlIHtcbiAgICBib3JkZXItY29sb3I6IHRyYW5zcGFyZW50O1xuICB9XG4gIGJhY2tncm91bmQtY29sb3I6ICR7KHsgdGhlbWUgfSkgPT5cbiAgICB0aGVtZS5jb2xvcnMub3RoZXIuZWxldmF0aW9uLmJhY2tncm91bmQucmFpc2VkfTtcbiAgY29sb3I6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuY29sb3JzLm5ldXRyYWwudGV4dH07XG4gIGJvcmRlci1yYWRpdXM6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUucmFkaWkuZGVmYXVsdH07XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbmBcblxuY29uc3QgU3R5bGVkU2VhcmNoSW5wdXQgPSBzdHlsZWQoU2VhcmNoSW5wdXQpYFxuICBwYWRkaW5nOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLnNwYWNlWycxJ119O1xuYFxuXG5leHBvcnQgY29uc3QgTWVudSA9IGZvcndhcmRSZWYoXG4gIChcbiAgICB7XG4gICAgICBpZCxcbiAgICAgIGFyaWFMYWJlbCA9ICdNZW51JyxcbiAgICAgIGNoaWxkcmVuLFxuICAgICAgZGlzY2xvc3VyZSxcbiAgICAgIGhhc0Fycm93ID0gZmFsc2UsXG4gICAgICBwbGFjZW1lbnQgPSAnYm90dG9tJyxcbiAgICAgIGNsYXNzTmFtZSxcbiAgICAgICdkYXRhLXRlc3RpZCc6IGRhdGFUZXN0SWQsXG4gICAgICBtYXhIZWlnaHQsXG4gICAgICBtYXhXaWR0aCxcbiAgICAgIHBvcnRhbFRhcmdldCA9IGRvY3VtZW50LmJvZHksXG4gICAgICBzaXplID0gJ3NtYWxsJyxcbiAgICAgIHRyaWdnZXJNZXRob2QgPSAnY2xpY2snLFxuICAgICAgZHluYW1pY0RvbVJlbmRlcmluZyxcbiAgICAgIGFsaWduLFxuICAgICAgc2VhcmNoYWJsZSA9IGZhbHNlLFxuICAgIH06IE1lbnVQcm9wcyxcbiAgICByZWY6IFJlZjxIVE1MQnV0dG9uRWxlbWVudCB8IG51bGw+LFxuICApID0+IHtcbiAgICBjb25zdCB7IGlzVmlzaWJsZSwgc2V0SXNWaXNpYmxlIH0gPSB1c2VNZW51KClcbiAgICBjb25zdCBzZWFyY2hJbnB1dFJlZiA9IHVzZVJlZjxIVE1MSW5wdXRFbGVtZW50PihudWxsKVxuICAgIGNvbnN0IFtsb2NhbENoaWxkLCBzZXRMb2NhbENoaWxkXSA9IHVzZVN0YXRlPFJlYWN0Tm9kZVtdIHwgbnVsbD4obnVsbClcbiAgICBjb25zdCBwb3B1cFJlZiA9IHVzZVJlZjxIVE1MRGl2RWxlbWVudD4obnVsbClcbiAgICBjb25zdCBkaXNjbG9zdXJlUmVmID0gdXNlUmVmPEhUTUxCdXR0b25FbGVtZW50PihudWxsKVxuICAgIGNvbnN0IHRlbXBJZCA9IHVzZUlkKClcbiAgICBjb25zdCBmaW5hbElkID0gYG1lbnUtJHtpZCA/PyB0ZW1wSWR9YFxuXG4gICAgLy8gaWYgeW91IG5lZWQgZGlhbG9nIGluc2lkZSB5b3VyIGNvbXBvbmVudCwgdXNlIGZ1bmN0aW9uLCBvdGhlcndpc2UgY29tcG9uZW50IGlzIGZpbmVcbiAgICBjb25zdCB0YXJnZXQgPSBpc1ZhbGlkRWxlbWVudDxCdXR0b25IVE1MQXR0cmlidXRlczxIVE1MQnV0dG9uRWxlbWVudD4+KFxuICAgICAgZGlzY2xvc3VyZSxcbiAgICApXG4gICAgICA/IGRpc2Nsb3N1cmVcbiAgICAgIDogZGlzY2xvc3VyZSh7IHZpc2libGU6IGlzVmlzaWJsZSB9KVxuICAgIGNvbnN0IGlubmVyUmVmID0gdXNlUmVmKHRhcmdldCBhcyB1bmtub3duIGFzIEhUTUxCdXR0b25FbGVtZW50KVxuICAgIHVzZUltcGVyYXRpdmVIYW5kbGUocmVmLCAoKSA9PiBpbm5lclJlZi5jdXJyZW50KVxuXG4gICAgY29uc3QgZmluYWxEaXNjbG9zdXJlID0gY2xvbmVFbGVtZW50KHRhcmdldCwge1xuICAgICAgb25DbGljazogKGV2ZW50OiBNb3VzZUV2ZW50PEhUTUxCdXR0b25FbGVtZW50PikgPT4ge1xuICAgICAgICB0YXJnZXQucHJvcHMub25DbGljaz8uKGV2ZW50KVxuICAgICAgICBzZXRJc1Zpc2libGUoIWlzVmlzaWJsZSlcbiAgICAgIH0sXG4gICAgICAnYXJpYS1oYXNwb3B1cCc6ICdkaWFsb2cnLFxuICAgICAgJ2FyaWEtZXhwYW5kZWQnOiBpc1Zpc2libGUsXG4gICAgICAvLyBAdHMtZXhwZWN0LWVycm9yIG5vdCBzdXJlIGhvdyB0byBmaXggdGhpc1xuICAgICAgcmVmOiBkaXNjbG9zdXJlUmVmLFxuICAgIH0pXG5cbiAgICBjb25zdCBvblNlYXJjaCA9IHVzZUNhbGxiYWNrKFxuICAgICAgKHZhbHVlOiBzdHJpbmcpID0+IHtcbiAgICAgICAgaWYgKHR5cGVvZiBjaGlsZHJlbiA9PT0gJ29iamVjdCcpIHtcbiAgICAgICAgICBzZXRMb2NhbENoaWxkKHNlYXJjaENoaWxkcmVuKGNoaWxkcmVuLCB2YWx1ZSkpXG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgICBbY2hpbGRyZW5dLFxuICAgIClcblxuICAgIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgICBpZiAoaXNWaXNpYmxlICYmIHNlYXJjaGFibGUpIHtcbiAgICAgICAgc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICAgICAgc2VhcmNoSW5wdXRSZWYuY3VycmVudD8uZm9jdXMoKVxuICAgICAgICB9LCA1MClcbiAgICAgIH1cbiAgICB9LCBbaXNWaXNpYmxlLCBzZWFyY2hhYmxlXSlcblxuICAgIGNvbnN0IGZpbmFsQ2hpbGQgPSB1c2VNZW1vKCgpID0+IHtcbiAgICAgIGlmICh0eXBlb2YgY2hpbGRyZW4gPT09ICdmdW5jdGlvbicpIHtcbiAgICAgICAgcmV0dXJuIGNoaWxkcmVuKHsgdG9nZ2xlOiAoKSA9PiBzZXRJc1Zpc2libGUoIWlzVmlzaWJsZSkgfSlcbiAgICAgIH1cblxuICAgICAgaWYgKHNlYXJjaGFibGUgJiYgbG9jYWxDaGlsZCkge1xuICAgICAgICByZXR1cm4gbG9jYWxDaGlsZFxuICAgICAgfVxuXG4gICAgICByZXR1cm4gY2hpbGRyZW5cbiAgICB9LCBbY2hpbGRyZW4sIGlzVmlzaWJsZSwgbG9jYWxDaGlsZCwgc2VhcmNoYWJsZSwgc2V0SXNWaXNpYmxlXSlcblxuICAgIHJldHVybiAoXG4gICAgICA8U3R5bGVkUG9wdXBcbiAgICAgICAgZGVib3VuY2VEZWxheT17dHJpZ2dlck1ldGhvZCA9PT0gJ2hvdmVyJyA/IDI1MCA6IDB9XG4gICAgICAgIGhpZGVPbkNsaWNrT3V0c2lkZVxuICAgICAgICBhcmlhLWxhYmVsPXthcmlhTGFiZWx9XG4gICAgICAgIGNsYXNzTmFtZT17Y2xhc3NOYW1lfVxuICAgICAgICB2aXNpYmxlPXt0cmlnZ2VyTWV0aG9kID09PSAnY2xpY2snID8gaXNWaXNpYmxlIDogdW5kZWZpbmVkfVxuICAgICAgICBwbGFjZW1lbnQ9e3BsYWNlbWVudH1cbiAgICAgICAgaGFzQXJyb3c9e2hhc0Fycm93fVxuICAgICAgICBkYXRhLWhhcy1hcnJvdz17aGFzQXJyb3d9XG4gICAgICAgIHJvbGU9XCJkaWFsb2dcIlxuICAgICAgICBpZD17ZmluYWxJZH1cbiAgICAgICAgcmVmPXtwb3B1cFJlZn1cbiAgICAgICAgb25DbG9zZT17KCkgPT4ge1xuICAgICAgICAgIHNldElzVmlzaWJsZShmYWxzZSlcbiAgICAgICAgICBzZXRMb2NhbENoaWxkKG51bGwpXG4gICAgICAgIH19XG4gICAgICAgIHRhYkluZGV4PXstMX1cbiAgICAgICAgbWF4SGVpZ2h0PXttYXhIZWlnaHQgPz8gJzMwcmVtJ31cbiAgICAgICAgbWF4V2lkdGg9e21heFdpZHRofVxuICAgICAgICBzZWFyY2hhYmxlPXtzZWFyY2hhYmxlfVxuICAgICAgICBzaXplPXtzaXplfVxuICAgICAgICB0ZXh0PXtcbiAgICAgICAgICA8ZGl2PlxuICAgICAgICAgICAge3NlYXJjaGFibGUgJiYgdHlwZW9mIGNoaWxkcmVuICE9PSAnZnVuY3Rpb24nID8gKFxuICAgICAgICAgICAgICA8U3R5bGVkU2VhcmNoSW5wdXRcbiAgICAgICAgICAgICAgICBzaXplPVwic21hbGxcIlxuICAgICAgICAgICAgICAgIG9uU2VhcmNoPXtvblNlYXJjaH1cbiAgICAgICAgICAgICAgICByZWY9e3NlYXJjaElucHV0UmVmfVxuICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgKSA6IG51bGx9XG4gICAgICAgICAgICA8TWVudUxpc3RcbiAgICAgICAgICAgICAgZGF0YS10ZXN0aWQ9e2RhdGFUZXN0SWR9XG4gICAgICAgICAgICAgIGNsYXNzTmFtZT17Y2xhc3NOYW1lfVxuICAgICAgICAgICAgICByb2xlPVwibWVudVwiXG4gICAgICAgICAgICA+XG4gICAgICAgICAgICAgIHtmaW5hbENoaWxkfVxuICAgICAgICAgICAgPC9NZW51TGlzdD5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgfVxuICAgICAgICBwb3J0YWxUYXJnZXQ9e3BvcnRhbFRhcmdldH1cbiAgICAgICAgZHluYW1pY0RvbVJlbmRlcmluZz17ZHluYW1pY0RvbVJlbmRlcmluZ31cbiAgICAgICAgYWxpZ249e2FsaWdufVxuICAgICAgPlxuICAgICAgICB7ZmluYWxEaXNjbG9zdXJlfVxuICAgICAgPC9TdHlsZWRQb3B1cD5cbiAgICApXG4gIH0sXG4pXG4iXX0= */"));
29
+ }) => `${theme.space["0.25"]} 0`, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/MenuContent.tsx"],"names":[],"mappings":"AAwBsD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/MenuContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type { ButtonHTMLAttributes, MouseEvent, ReactNode, Ref } from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { SearchInput } from '../SearchInput'\nimport { Stack } from '../Stack'\nimport { useMenu } from './MenuProvider'\nimport { SIZES } from './constants'\nimport { searchChildren } from './helpers'\nimport type { MenuProps } from './types'\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size', 'searchable'].includes(prop),\n})<{ size?: keyof typeof SIZES; searchable: boolean }>`\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) => theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  min-width: ${SIZES.small};\n  max-width: ${SIZES.large};\n  ${({ size }) => (size ? `width: ${SIZES[size]};` : null)}\n  ${({ searchable }) => (searchable ? `min-width: 20rem` : null)};\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n`\n\nconst MenuList = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\nconst StyledSearchInput = styled(SearchInput)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nexport const Menu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget = document.body,\n      size,\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n      searchable = false,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const { isVisible, setIsVisible } = useMenu()\n    const searchInputRef = useRef<HTMLInputElement>(null)\n    const [localChild, setLocalChild] = useState<ReactNode[] | null>(null)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const disclosureRef = useRef<HTMLButtonElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      'aria-haspopup': 'dialog',\n      'aria-expanded': isVisible,\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    const onSearch = useCallback(\n      (value: string) => {\n        if (typeof children === 'object') {\n          setLocalChild(searchChildren(children, value))\n        }\n      },\n      [children],\n    )\n\n    useEffect(() => {\n      if (isVisible && searchable) {\n        setTimeout(() => {\n          searchInputRef.current?.focus()\n        }, 50)\n      }\n    }, [isVisible, searchable])\n\n    const finalChild = useMemo(() => {\n      if (typeof children === 'function') {\n        return children({ toggle: () => setIsVisible(!isVisible) })\n      }\n\n      if (searchable && localChild) {\n        return localChild\n      }\n\n      return children\n    }, [children, isVisible, localChild, searchable, setIsVisible])\n\n    return (\n      <StyledPopup\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        hideOnClickOutside\n        aria-label={ariaLabel}\n        className={className}\n        visible={triggerMethod === 'click' ? isVisible : undefined}\n        placement={placement}\n        hasArrow={hasArrow}\n        data-has-arrow={hasArrow}\n        role=\"dialog\"\n        id={finalId}\n        ref={popupRef}\n        onClose={() => {\n          setIsVisible(false)\n          setLocalChild(null)\n        }}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '30rem'}\n        maxWidth={maxWidth}\n        searchable={searchable}\n        size={size}\n        text={\n          <div>\n            {searchable && typeof children !== 'function' ? (\n              <StyledSearchInput\n                size=\"small\"\n                onSearch={onSearch}\n                ref={searchInputRef}\n              />\n            ) : null}\n            <MenuList\n              data-testid={dataTestId}\n              className={className}\n              role=\"menu\"\n            >\n              {finalChild}\n            </MenuList>\n          </div>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n        align={align}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n"]} */"));
30
30
  const MenuList = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
31
31
  target: "e1rdim8w1"
32
32
  } : {
@@ -38,7 +38,7 @@ const MenuList = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "produc
38
38
  theme
39
39
  }) => theme.colors.neutral.text, ";border-radius:", ({
40
40
  theme
41
- }) => theme.radii.default, ";position:relative;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01lbnVWMi9NZW51Q29udGVudC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBMEM4QiIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9NZW51VjIvTWVudUNvbnRlbnQudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7IEJ1dHRvbkhUTUxBdHRyaWJ1dGVzLCBNb3VzZUV2ZW50LCBSZWFjdE5vZGUsIFJlZiB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHtcbiAgY2xvbmVFbGVtZW50LFxuICBmb3J3YXJkUmVmLFxuICBpc1ZhbGlkRWxlbWVudCxcbiAgdXNlQ2FsbGJhY2ssXG4gIHVzZUVmZmVjdCxcbiAgdXNlSWQsXG4gIHVzZUltcGVyYXRpdmVIYW5kbGUsXG4gIHVzZU1lbW8sXG4gIHVzZVJlZixcbiAgdXNlU3RhdGUsXG59IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgUG9wdXAgfSBmcm9tICcuLi9Qb3B1cCdcbmltcG9ydCB7IFNlYXJjaElucHV0IH0gZnJvbSAnLi4vU2VhcmNoSW5wdXQnXG5pbXBvcnQgeyBTdGFjayB9IGZyb20gJy4uL1N0YWNrJ1xuaW1wb3J0IHsgdXNlTWVudSB9IGZyb20gJy4vTWVudVByb3ZpZGVyJ1xuaW1wb3J0IHsgU0laRVMgfSBmcm9tICcuL2NvbnN0YW50cydcbmltcG9ydCB7IHNlYXJjaENoaWxkcmVuIH0gZnJvbSAnLi9oZWxwZXJzJ1xuaW1wb3J0IHR5cGUgeyBNZW51UHJvcHMgfSBmcm9tICcuL3R5cGVzJ1xuXG5jb25zdCBTdHlsZWRQb3B1cCA9IHN0eWxlZChQb3B1cCwge1xuICBzaG91bGRGb3J3YXJkUHJvcDogcHJvcCA9PiAhWydzaXplJywgJ3NlYXJjaGFibGUnXS5pbmNsdWRlcyhwcm9wKSxcbn0pPHsgc2l6ZToga2V5b2YgdHlwZW9mIFNJWkVTOyBzZWFyY2hhYmxlOiBib29sZWFuIH0+YFxuICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLmNvbG9ycy5vdGhlci5lbGV2YXRpb24uYmFja2dyb3VuZC5yYWlzZWR9O1xuICBib3gtc2hhZG93OiAkeyh7IHRoZW1lIH0pID0+IGAke3RoZW1lLnNoYWRvd3MucmFpc2VkWzBdfSwgJHt0aGVtZS5zaGFkb3dzLnJhaXNlZFsxXX1gfTtcbiAgcGFkZGluZzogMDtcblxuICAmW2RhdGEtaGFzLWFycm93PSd0cnVlJ10ge1xuICAgICY6OmFmdGVyIHtcbiAgICAgIGJvcmRlci1jb2xvcjogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5jb2xvcnMub3RoZXIuZWxldmF0aW9uLmJhY2tncm91bmQucmFpc2VkfVxuICAgICAgICB0cmFuc3BhcmVudCB0cmFuc3BhcmVudCB0cmFuc3BhcmVudDtcbiAgICB9XG4gIH1cblxuICB3aWR0aDogJHsoeyBzaXplIH0pID0+IFNJWkVTW3NpemVdfTtcbiAgbWF4LXdpZHRoOiBub25lO1xuICAkeyh7IHNlYXJjaGFibGUgfSkgPT4gKHNlYXJjaGFibGUgPyBgbWluLXdpZHRoOiAyMHJlbWAgOiBudWxsKX07XG4gIHBhZGRpbmc6ICR7KHsgdGhlbWUgfSkgPT4gYCR7dGhlbWUuc3BhY2VbJzAuMjUnXX0gMGB9O1xuYFxuXG5jb25zdCBNZW51TGlzdCA9IHN0eWxlZChTdGFjaylgXG4gIG92ZXJmbG93LXk6IGF1dG87XG4gIG92ZXJmbG93LXg6IGhpZGRlbjtcbiAgJjphZnRlcixcbiAgJjpiZWZvcmUge1xuICAgIGJvcmRlcjogc29saWQgdHJhbnNwYXJlbnQ7XG4gICAgYm9yZGVyLXdpZHRoOiA5cHg7XG4gICAgY29udGVudDogJyAnO1xuICAgIGhlaWdodDogMDtcbiAgICB3aWR0aDogMDtcbiAgICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gICAgcG9pbnRlci1ldmVudHM6IG5vbmU7XG4gIH1cblxuICAmOmFmdGVyIHtcbiAgICBib3JkZXItY29sb3I6IHRyYW5zcGFyZW50O1xuICB9XG4gICY6YmVmb3JlIHtcbiAgICBib3JkZXItY29sb3I6IHRyYW5zcGFyZW50O1xuICB9XG4gIGJhY2tncm91bmQtY29sb3I6ICR7KHsgdGhlbWUgfSkgPT5cbiAgICB0aGVtZS5jb2xvcnMub3RoZXIuZWxldmF0aW9uLmJhY2tncm91bmQucmFpc2VkfTtcbiAgY29sb3I6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuY29sb3JzLm5ldXRyYWwudGV4dH07XG4gIGJvcmRlci1yYWRpdXM6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUucmFkaWkuZGVmYXVsdH07XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbmBcblxuY29uc3QgU3R5bGVkU2VhcmNoSW5wdXQgPSBzdHlsZWQoU2VhcmNoSW5wdXQpYFxuICBwYWRkaW5nOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLnNwYWNlWycxJ119O1xuYFxuXG5leHBvcnQgY29uc3QgTWVudSA9IGZvcndhcmRSZWYoXG4gIChcbiAgICB7XG4gICAgICBpZCxcbiAgICAgIGFyaWFMYWJlbCA9ICdNZW51JyxcbiAgICAgIGNoaWxkcmVuLFxuICAgICAgZGlzY2xvc3VyZSxcbiAgICAgIGhhc0Fycm93ID0gZmFsc2UsXG4gICAgICBwbGFjZW1lbnQgPSAnYm90dG9tJyxcbiAgICAgIGNsYXNzTmFtZSxcbiAgICAgICdkYXRhLXRlc3RpZCc6IGRhdGFUZXN0SWQsXG4gICAgICBtYXhIZWlnaHQsXG4gICAgICBtYXhXaWR0aCxcbiAgICAgIHBvcnRhbFRhcmdldCA9IGRvY3VtZW50LmJvZHksXG4gICAgICBzaXplID0gJ3NtYWxsJyxcbiAgICAgIHRyaWdnZXJNZXRob2QgPSAnY2xpY2snLFxuICAgICAgZHluYW1pY0RvbVJlbmRlcmluZyxcbiAgICAgIGFsaWduLFxuICAgICAgc2VhcmNoYWJsZSA9IGZhbHNlLFxuICAgIH06IE1lbnVQcm9wcyxcbiAgICByZWY6IFJlZjxIVE1MQnV0dG9uRWxlbWVudCB8IG51bGw+LFxuICApID0+IHtcbiAgICBjb25zdCB7IGlzVmlzaWJsZSwgc2V0SXNWaXNpYmxlIH0gPSB1c2VNZW51KClcbiAgICBjb25zdCBzZWFyY2hJbnB1dFJlZiA9IHVzZVJlZjxIVE1MSW5wdXRFbGVtZW50PihudWxsKVxuICAgIGNvbnN0IFtsb2NhbENoaWxkLCBzZXRMb2NhbENoaWxkXSA9IHVzZVN0YXRlPFJlYWN0Tm9kZVtdIHwgbnVsbD4obnVsbClcbiAgICBjb25zdCBwb3B1cFJlZiA9IHVzZVJlZjxIVE1MRGl2RWxlbWVudD4obnVsbClcbiAgICBjb25zdCBkaXNjbG9zdXJlUmVmID0gdXNlUmVmPEhUTUxCdXR0b25FbGVtZW50PihudWxsKVxuICAgIGNvbnN0IHRlbXBJZCA9IHVzZUlkKClcbiAgICBjb25zdCBmaW5hbElkID0gYG1lbnUtJHtpZCA/PyB0ZW1wSWR9YFxuXG4gICAgLy8gaWYgeW91IG5lZWQgZGlhbG9nIGluc2lkZSB5b3VyIGNvbXBvbmVudCwgdXNlIGZ1bmN0aW9uLCBvdGhlcndpc2UgY29tcG9uZW50IGlzIGZpbmVcbiAgICBjb25zdCB0YXJnZXQgPSBpc1ZhbGlkRWxlbWVudDxCdXR0b25IVE1MQXR0cmlidXRlczxIVE1MQnV0dG9uRWxlbWVudD4+KFxuICAgICAgZGlzY2xvc3VyZSxcbiAgICApXG4gICAgICA/IGRpc2Nsb3N1cmVcbiAgICAgIDogZGlzY2xvc3VyZSh7IHZpc2libGU6IGlzVmlzaWJsZSB9KVxuICAgIGNvbnN0IGlubmVyUmVmID0gdXNlUmVmKHRhcmdldCBhcyB1bmtub3duIGFzIEhUTUxCdXR0b25FbGVtZW50KVxuICAgIHVzZUltcGVyYXRpdmVIYW5kbGUocmVmLCAoKSA9PiBpbm5lclJlZi5jdXJyZW50KVxuXG4gICAgY29uc3QgZmluYWxEaXNjbG9zdXJlID0gY2xvbmVFbGVtZW50KHRhcmdldCwge1xuICAgICAgb25DbGljazogKGV2ZW50OiBNb3VzZUV2ZW50PEhUTUxCdXR0b25FbGVtZW50PikgPT4ge1xuICAgICAgICB0YXJnZXQucHJvcHMub25DbGljaz8uKGV2ZW50KVxuICAgICAgICBzZXRJc1Zpc2libGUoIWlzVmlzaWJsZSlcbiAgICAgIH0sXG4gICAgICAnYXJpYS1oYXNwb3B1cCc6ICdkaWFsb2cnLFxuICAgICAgJ2FyaWEtZXhwYW5kZWQnOiBpc1Zpc2libGUsXG4gICAgICAvLyBAdHMtZXhwZWN0LWVycm9yIG5vdCBzdXJlIGhvdyB0byBmaXggdGhpc1xuICAgICAgcmVmOiBkaXNjbG9zdXJlUmVmLFxuICAgIH0pXG5cbiAgICBjb25zdCBvblNlYXJjaCA9IHVzZUNhbGxiYWNrKFxuICAgICAgKHZhbHVlOiBzdHJpbmcpID0+IHtcbiAgICAgICAgaWYgKHR5cGVvZiBjaGlsZHJlbiA9PT0gJ29iamVjdCcpIHtcbiAgICAgICAgICBzZXRMb2NhbENoaWxkKHNlYXJjaENoaWxkcmVuKGNoaWxkcmVuLCB2YWx1ZSkpXG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgICBbY2hpbGRyZW5dLFxuICAgIClcblxuICAgIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgICBpZiAoaXNWaXNpYmxlICYmIHNlYXJjaGFibGUpIHtcbiAgICAgICAgc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICAgICAgc2VhcmNoSW5wdXRSZWYuY3VycmVudD8uZm9jdXMoKVxuICAgICAgICB9LCA1MClcbiAgICAgIH1cbiAgICB9LCBbaXNWaXNpYmxlLCBzZWFyY2hhYmxlXSlcblxuICAgIGNvbnN0IGZpbmFsQ2hpbGQgPSB1c2VNZW1vKCgpID0+IHtcbiAgICAgIGlmICh0eXBlb2YgY2hpbGRyZW4gPT09ICdmdW5jdGlvbicpIHtcbiAgICAgICAgcmV0dXJuIGNoaWxkcmVuKHsgdG9nZ2xlOiAoKSA9PiBzZXRJc1Zpc2libGUoIWlzVmlzaWJsZSkgfSlcbiAgICAgIH1cblxuICAgICAgaWYgKHNlYXJjaGFibGUgJiYgbG9jYWxDaGlsZCkge1xuICAgICAgICByZXR1cm4gbG9jYWxDaGlsZFxuICAgICAgfVxuXG4gICAgICByZXR1cm4gY2hpbGRyZW5cbiAgICB9LCBbY2hpbGRyZW4sIGlzVmlzaWJsZSwgbG9jYWxDaGlsZCwgc2VhcmNoYWJsZSwgc2V0SXNWaXNpYmxlXSlcblxuICAgIHJldHVybiAoXG4gICAgICA8U3R5bGVkUG9wdXBcbiAgICAgICAgZGVib3VuY2VEZWxheT17dHJpZ2dlck1ldGhvZCA9PT0gJ2hvdmVyJyA/IDI1MCA6IDB9XG4gICAgICAgIGhpZGVPbkNsaWNrT3V0c2lkZVxuICAgICAgICBhcmlhLWxhYmVsPXthcmlhTGFiZWx9XG4gICAgICAgIGNsYXNzTmFtZT17Y2xhc3NOYW1lfVxuICAgICAgICB2aXNpYmxlPXt0cmlnZ2VyTWV0aG9kID09PSAnY2xpY2snID8gaXNWaXNpYmxlIDogdW5kZWZpbmVkfVxuICAgICAgICBwbGFjZW1lbnQ9e3BsYWNlbWVudH1cbiAgICAgICAgaGFzQXJyb3c9e2hhc0Fycm93fVxuICAgICAgICBkYXRhLWhhcy1hcnJvdz17aGFzQXJyb3d9XG4gICAgICAgIHJvbGU9XCJkaWFsb2dcIlxuICAgICAgICBpZD17ZmluYWxJZH1cbiAgICAgICAgcmVmPXtwb3B1cFJlZn1cbiAgICAgICAgb25DbG9zZT17KCkgPT4ge1xuICAgICAgICAgIHNldElzVmlzaWJsZShmYWxzZSlcbiAgICAgICAgICBzZXRMb2NhbENoaWxkKG51bGwpXG4gICAgICAgIH19XG4gICAgICAgIHRhYkluZGV4PXstMX1cbiAgICAgICAgbWF4SGVpZ2h0PXttYXhIZWlnaHQgPz8gJzMwcmVtJ31cbiAgICAgICAgbWF4V2lkdGg9e21heFdpZHRofVxuICAgICAgICBzZWFyY2hhYmxlPXtzZWFyY2hhYmxlfVxuICAgICAgICBzaXplPXtzaXplfVxuICAgICAgICB0ZXh0PXtcbiAgICAgICAgICA8ZGl2PlxuICAgICAgICAgICAge3NlYXJjaGFibGUgJiYgdHlwZW9mIGNoaWxkcmVuICE9PSAnZnVuY3Rpb24nID8gKFxuICAgICAgICAgICAgICA8U3R5bGVkU2VhcmNoSW5wdXRcbiAgICAgICAgICAgICAgICBzaXplPVwic21hbGxcIlxuICAgICAgICAgICAgICAgIG9uU2VhcmNoPXtvblNlYXJjaH1cbiAgICAgICAgICAgICAgICByZWY9e3NlYXJjaElucHV0UmVmfVxuICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgKSA6IG51bGx9XG4gICAgICAgICAgICA8TWVudUxpc3RcbiAgICAgICAgICAgICAgZGF0YS10ZXN0aWQ9e2RhdGFUZXN0SWR9XG4gICAgICAgICAgICAgIGNsYXNzTmFtZT17Y2xhc3NOYW1lfVxuICAgICAgICAgICAgICByb2xlPVwibWVudVwiXG4gICAgICAgICAgICA+XG4gICAgICAgICAgICAgIHtmaW5hbENoaWxkfVxuICAgICAgICAgICAgPC9NZW51TGlzdD5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgfVxuICAgICAgICBwb3J0YWxUYXJnZXQ9e3BvcnRhbFRhcmdldH1cbiAgICAgICAgZHluYW1pY0RvbVJlbmRlcmluZz17ZHluYW1pY0RvbVJlbmRlcmluZ31cbiAgICAgICAgYWxpZ249e2FsaWdufVxuICAgICAgPlxuICAgICAgICB7ZmluYWxEaXNjbG9zdXJlfVxuICAgICAgPC9TdHlsZWRQb3B1cD5cbiAgICApXG4gIH0sXG4pXG4iXX0= */"));
41
+ }) => theme.radii.default, ";position:relative;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/MenuContent.tsx"],"names":[],"mappings":"AA2C8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/MenuContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type { ButtonHTMLAttributes, MouseEvent, ReactNode, Ref } from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { SearchInput } from '../SearchInput'\nimport { Stack } from '../Stack'\nimport { useMenu } from './MenuProvider'\nimport { SIZES } from './constants'\nimport { searchChildren } from './helpers'\nimport type { MenuProps } from './types'\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size', 'searchable'].includes(prop),\n})<{ size?: keyof typeof SIZES; searchable: boolean }>`\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) => theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  min-width: ${SIZES.small};\n  max-width: ${SIZES.large};\n  ${({ size }) => (size ? `width: ${SIZES[size]};` : null)}\n  ${({ searchable }) => (searchable ? `min-width: 20rem` : null)};\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n`\n\nconst MenuList = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\nconst StyledSearchInput = styled(SearchInput)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nexport const Menu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget = document.body,\n      size,\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n      searchable = false,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const { isVisible, setIsVisible } = useMenu()\n    const searchInputRef = useRef<HTMLInputElement>(null)\n    const [localChild, setLocalChild] = useState<ReactNode[] | null>(null)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const disclosureRef = useRef<HTMLButtonElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      'aria-haspopup': 'dialog',\n      'aria-expanded': isVisible,\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    const onSearch = useCallback(\n      (value: string) => {\n        if (typeof children === 'object') {\n          setLocalChild(searchChildren(children, value))\n        }\n      },\n      [children],\n    )\n\n    useEffect(() => {\n      if (isVisible && searchable) {\n        setTimeout(() => {\n          searchInputRef.current?.focus()\n        }, 50)\n      }\n    }, [isVisible, searchable])\n\n    const finalChild = useMemo(() => {\n      if (typeof children === 'function') {\n        return children({ toggle: () => setIsVisible(!isVisible) })\n      }\n\n      if (searchable && localChild) {\n        return localChild\n      }\n\n      return children\n    }, [children, isVisible, localChild, searchable, setIsVisible])\n\n    return (\n      <StyledPopup\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        hideOnClickOutside\n        aria-label={ariaLabel}\n        className={className}\n        visible={triggerMethod === 'click' ? isVisible : undefined}\n        placement={placement}\n        hasArrow={hasArrow}\n        data-has-arrow={hasArrow}\n        role=\"dialog\"\n        id={finalId}\n        ref={popupRef}\n        onClose={() => {\n          setIsVisible(false)\n          setLocalChild(null)\n        }}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '30rem'}\n        maxWidth={maxWidth}\n        searchable={searchable}\n        size={size}\n        text={\n          <div>\n            {searchable && typeof children !== 'function' ? (\n              <StyledSearchInput\n                size=\"small\"\n                onSearch={onSearch}\n                ref={searchInputRef}\n              />\n            ) : null}\n            <MenuList\n              data-testid={dataTestId}\n              className={className}\n              role=\"menu\"\n            >\n              {finalChild}\n            </MenuList>\n          </div>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n        align={align}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n"]} */"));
42
42
  const StyledSearchInput = /* @__PURE__ */ _styled(SearchInput, process.env.NODE_ENV === "production" ? {
43
43
  target: "e1rdim8w0"
44
44
  } : {
@@ -46,7 +46,7 @@ const StyledSearchInput = /* @__PURE__ */ _styled(SearchInput, process.env.NODE_
46
46
  label: "StyledSearchInput"
47
47
  })("padding:", ({
48
48
  theme
49
- }) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01lbnVWMi9NZW51Q29udGVudC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBcUU2QyIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9NZW51VjIvTWVudUNvbnRlbnQudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7IEJ1dHRvbkhUTUxBdHRyaWJ1dGVzLCBNb3VzZUV2ZW50LCBSZWFjdE5vZGUsIFJlZiB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHtcbiAgY2xvbmVFbGVtZW50LFxuICBmb3J3YXJkUmVmLFxuICBpc1ZhbGlkRWxlbWVudCxcbiAgdXNlQ2FsbGJhY2ssXG4gIHVzZUVmZmVjdCxcbiAgdXNlSWQsXG4gIHVzZUltcGVyYXRpdmVIYW5kbGUsXG4gIHVzZU1lbW8sXG4gIHVzZVJlZixcbiAgdXNlU3RhdGUsXG59IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgUG9wdXAgfSBmcm9tICcuLi9Qb3B1cCdcbmltcG9ydCB7IFNlYXJjaElucHV0IH0gZnJvbSAnLi4vU2VhcmNoSW5wdXQnXG5pbXBvcnQgeyBTdGFjayB9IGZyb20gJy4uL1N0YWNrJ1xuaW1wb3J0IHsgdXNlTWVudSB9IGZyb20gJy4vTWVudVByb3ZpZGVyJ1xuaW1wb3J0IHsgU0laRVMgfSBmcm9tICcuL2NvbnN0YW50cydcbmltcG9ydCB7IHNlYXJjaENoaWxkcmVuIH0gZnJvbSAnLi9oZWxwZXJzJ1xuaW1wb3J0IHR5cGUgeyBNZW51UHJvcHMgfSBmcm9tICcuL3R5cGVzJ1xuXG5jb25zdCBTdHlsZWRQb3B1cCA9IHN0eWxlZChQb3B1cCwge1xuICBzaG91bGRGb3J3YXJkUHJvcDogcHJvcCA9PiAhWydzaXplJywgJ3NlYXJjaGFibGUnXS5pbmNsdWRlcyhwcm9wKSxcbn0pPHsgc2l6ZToga2V5b2YgdHlwZW9mIFNJWkVTOyBzZWFyY2hhYmxlOiBib29sZWFuIH0+YFxuICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLmNvbG9ycy5vdGhlci5lbGV2YXRpb24uYmFja2dyb3VuZC5yYWlzZWR9O1xuICBib3gtc2hhZG93OiAkeyh7IHRoZW1lIH0pID0+IGAke3RoZW1lLnNoYWRvd3MucmFpc2VkWzBdfSwgJHt0aGVtZS5zaGFkb3dzLnJhaXNlZFsxXX1gfTtcbiAgcGFkZGluZzogMDtcblxuICAmW2RhdGEtaGFzLWFycm93PSd0cnVlJ10ge1xuICAgICY6OmFmdGVyIHtcbiAgICAgIGJvcmRlci1jb2xvcjogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5jb2xvcnMub3RoZXIuZWxldmF0aW9uLmJhY2tncm91bmQucmFpc2VkfVxuICAgICAgICB0cmFuc3BhcmVudCB0cmFuc3BhcmVudCB0cmFuc3BhcmVudDtcbiAgICB9XG4gIH1cblxuICB3aWR0aDogJHsoeyBzaXplIH0pID0+IFNJWkVTW3NpemVdfTtcbiAgbWF4LXdpZHRoOiBub25lO1xuICAkeyh7IHNlYXJjaGFibGUgfSkgPT4gKHNlYXJjaGFibGUgPyBgbWluLXdpZHRoOiAyMHJlbWAgOiBudWxsKX07XG4gIHBhZGRpbmc6ICR7KHsgdGhlbWUgfSkgPT4gYCR7dGhlbWUuc3BhY2VbJzAuMjUnXX0gMGB9O1xuYFxuXG5jb25zdCBNZW51TGlzdCA9IHN0eWxlZChTdGFjaylgXG4gIG92ZXJmbG93LXk6IGF1dG87XG4gIG92ZXJmbG93LXg6IGhpZGRlbjtcbiAgJjphZnRlcixcbiAgJjpiZWZvcmUge1xuICAgIGJvcmRlcjogc29saWQgdHJhbnNwYXJlbnQ7XG4gICAgYm9yZGVyLXdpZHRoOiA5cHg7XG4gICAgY29udGVudDogJyAnO1xuICAgIGhlaWdodDogMDtcbiAgICB3aWR0aDogMDtcbiAgICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gICAgcG9pbnRlci1ldmVudHM6IG5vbmU7XG4gIH1cblxuICAmOmFmdGVyIHtcbiAgICBib3JkZXItY29sb3I6IHRyYW5zcGFyZW50O1xuICB9XG4gICY6YmVmb3JlIHtcbiAgICBib3JkZXItY29sb3I6IHRyYW5zcGFyZW50O1xuICB9XG4gIGJhY2tncm91bmQtY29sb3I6ICR7KHsgdGhlbWUgfSkgPT5cbiAgICB0aGVtZS5jb2xvcnMub3RoZXIuZWxldmF0aW9uLmJhY2tncm91bmQucmFpc2VkfTtcbiAgY29sb3I6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuY29sb3JzLm5ldXRyYWwudGV4dH07XG4gIGJvcmRlci1yYWRpdXM6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUucmFkaWkuZGVmYXVsdH07XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbmBcblxuY29uc3QgU3R5bGVkU2VhcmNoSW5wdXQgPSBzdHlsZWQoU2VhcmNoSW5wdXQpYFxuICBwYWRkaW5nOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLnNwYWNlWycxJ119O1xuYFxuXG5leHBvcnQgY29uc3QgTWVudSA9IGZvcndhcmRSZWYoXG4gIChcbiAgICB7XG4gICAgICBpZCxcbiAgICAgIGFyaWFMYWJlbCA9ICdNZW51JyxcbiAgICAgIGNoaWxkcmVuLFxuICAgICAgZGlzY2xvc3VyZSxcbiAgICAgIGhhc0Fycm93ID0gZmFsc2UsXG4gICAgICBwbGFjZW1lbnQgPSAnYm90dG9tJyxcbiAgICAgIGNsYXNzTmFtZSxcbiAgICAgICdkYXRhLXRlc3RpZCc6IGRhdGFUZXN0SWQsXG4gICAgICBtYXhIZWlnaHQsXG4gICAgICBtYXhXaWR0aCxcbiAgICAgIHBvcnRhbFRhcmdldCA9IGRvY3VtZW50LmJvZHksXG4gICAgICBzaXplID0gJ3NtYWxsJyxcbiAgICAgIHRyaWdnZXJNZXRob2QgPSAnY2xpY2snLFxuICAgICAgZHluYW1pY0RvbVJlbmRlcmluZyxcbiAgICAgIGFsaWduLFxuICAgICAgc2VhcmNoYWJsZSA9IGZhbHNlLFxuICAgIH06IE1lbnVQcm9wcyxcbiAgICByZWY6IFJlZjxIVE1MQnV0dG9uRWxlbWVudCB8IG51bGw+LFxuICApID0+IHtcbiAgICBjb25zdCB7IGlzVmlzaWJsZSwgc2V0SXNWaXNpYmxlIH0gPSB1c2VNZW51KClcbiAgICBjb25zdCBzZWFyY2hJbnB1dFJlZiA9IHVzZVJlZjxIVE1MSW5wdXRFbGVtZW50PihudWxsKVxuICAgIGNvbnN0IFtsb2NhbENoaWxkLCBzZXRMb2NhbENoaWxkXSA9IHVzZVN0YXRlPFJlYWN0Tm9kZVtdIHwgbnVsbD4obnVsbClcbiAgICBjb25zdCBwb3B1cFJlZiA9IHVzZVJlZjxIVE1MRGl2RWxlbWVudD4obnVsbClcbiAgICBjb25zdCBkaXNjbG9zdXJlUmVmID0gdXNlUmVmPEhUTUxCdXR0b25FbGVtZW50PihudWxsKVxuICAgIGNvbnN0IHRlbXBJZCA9IHVzZUlkKClcbiAgICBjb25zdCBmaW5hbElkID0gYG1lbnUtJHtpZCA/PyB0ZW1wSWR9YFxuXG4gICAgLy8gaWYgeW91IG5lZWQgZGlhbG9nIGluc2lkZSB5b3VyIGNvbXBvbmVudCwgdXNlIGZ1bmN0aW9uLCBvdGhlcndpc2UgY29tcG9uZW50IGlzIGZpbmVcbiAgICBjb25zdCB0YXJnZXQgPSBpc1ZhbGlkRWxlbWVudDxCdXR0b25IVE1MQXR0cmlidXRlczxIVE1MQnV0dG9uRWxlbWVudD4+KFxuICAgICAgZGlzY2xvc3VyZSxcbiAgICApXG4gICAgICA/IGRpc2Nsb3N1cmVcbiAgICAgIDogZGlzY2xvc3VyZSh7IHZpc2libGU6IGlzVmlzaWJsZSB9KVxuICAgIGNvbnN0IGlubmVyUmVmID0gdXNlUmVmKHRhcmdldCBhcyB1bmtub3duIGFzIEhUTUxCdXR0b25FbGVtZW50KVxuICAgIHVzZUltcGVyYXRpdmVIYW5kbGUocmVmLCAoKSA9PiBpbm5lclJlZi5jdXJyZW50KVxuXG4gICAgY29uc3QgZmluYWxEaXNjbG9zdXJlID0gY2xvbmVFbGVtZW50KHRhcmdldCwge1xuICAgICAgb25DbGljazogKGV2ZW50OiBNb3VzZUV2ZW50PEhUTUxCdXR0b25FbGVtZW50PikgPT4ge1xuICAgICAgICB0YXJnZXQucHJvcHMub25DbGljaz8uKGV2ZW50KVxuICAgICAgICBzZXRJc1Zpc2libGUoIWlzVmlzaWJsZSlcbiAgICAgIH0sXG4gICAgICAnYXJpYS1oYXNwb3B1cCc6ICdkaWFsb2cnLFxuICAgICAgJ2FyaWEtZXhwYW5kZWQnOiBpc1Zpc2libGUsXG4gICAgICAvLyBAdHMtZXhwZWN0LWVycm9yIG5vdCBzdXJlIGhvdyB0byBmaXggdGhpc1xuICAgICAgcmVmOiBkaXNjbG9zdXJlUmVmLFxuICAgIH0pXG5cbiAgICBjb25zdCBvblNlYXJjaCA9IHVzZUNhbGxiYWNrKFxuICAgICAgKHZhbHVlOiBzdHJpbmcpID0+IHtcbiAgICAgICAgaWYgKHR5cGVvZiBjaGlsZHJlbiA9PT0gJ29iamVjdCcpIHtcbiAgICAgICAgICBzZXRMb2NhbENoaWxkKHNlYXJjaENoaWxkcmVuKGNoaWxkcmVuLCB2YWx1ZSkpXG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgICBbY2hpbGRyZW5dLFxuICAgIClcblxuICAgIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgICBpZiAoaXNWaXNpYmxlICYmIHNlYXJjaGFibGUpIHtcbiAgICAgICAgc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICAgICAgc2VhcmNoSW5wdXRSZWYuY3VycmVudD8uZm9jdXMoKVxuICAgICAgICB9LCA1MClcbiAgICAgIH1cbiAgICB9LCBbaXNWaXNpYmxlLCBzZWFyY2hhYmxlXSlcblxuICAgIGNvbnN0IGZpbmFsQ2hpbGQgPSB1c2VNZW1vKCgpID0+IHtcbiAgICAgIGlmICh0eXBlb2YgY2hpbGRyZW4gPT09ICdmdW5jdGlvbicpIHtcbiAgICAgICAgcmV0dXJuIGNoaWxkcmVuKHsgdG9nZ2xlOiAoKSA9PiBzZXRJc1Zpc2libGUoIWlzVmlzaWJsZSkgfSlcbiAgICAgIH1cblxuICAgICAgaWYgKHNlYXJjaGFibGUgJiYgbG9jYWxDaGlsZCkge1xuICAgICAgICByZXR1cm4gbG9jYWxDaGlsZFxuICAgICAgfVxuXG4gICAgICByZXR1cm4gY2hpbGRyZW5cbiAgICB9LCBbY2hpbGRyZW4sIGlzVmlzaWJsZSwgbG9jYWxDaGlsZCwgc2VhcmNoYWJsZSwgc2V0SXNWaXNpYmxlXSlcblxuICAgIHJldHVybiAoXG4gICAgICA8U3R5bGVkUG9wdXBcbiAgICAgICAgZGVib3VuY2VEZWxheT17dHJpZ2dlck1ldGhvZCA9PT0gJ2hvdmVyJyA/IDI1MCA6IDB9XG4gICAgICAgIGhpZGVPbkNsaWNrT3V0c2lkZVxuICAgICAgICBhcmlhLWxhYmVsPXthcmlhTGFiZWx9XG4gICAgICAgIGNsYXNzTmFtZT17Y2xhc3NOYW1lfVxuICAgICAgICB2aXNpYmxlPXt0cmlnZ2VyTWV0aG9kID09PSAnY2xpY2snID8gaXNWaXNpYmxlIDogdW5kZWZpbmVkfVxuICAgICAgICBwbGFjZW1lbnQ9e3BsYWNlbWVudH1cbiAgICAgICAgaGFzQXJyb3c9e2hhc0Fycm93fVxuICAgICAgICBkYXRhLWhhcy1hcnJvdz17aGFzQXJyb3d9XG4gICAgICAgIHJvbGU9XCJkaWFsb2dcIlxuICAgICAgICBpZD17ZmluYWxJZH1cbiAgICAgICAgcmVmPXtwb3B1cFJlZn1cbiAgICAgICAgb25DbG9zZT17KCkgPT4ge1xuICAgICAgICAgIHNldElzVmlzaWJsZShmYWxzZSlcbiAgICAgICAgICBzZXRMb2NhbENoaWxkKG51bGwpXG4gICAgICAgIH19XG4gICAgICAgIHRhYkluZGV4PXstMX1cbiAgICAgICAgbWF4SGVpZ2h0PXttYXhIZWlnaHQgPz8gJzMwcmVtJ31cbiAgICAgICAgbWF4V2lkdGg9e21heFdpZHRofVxuICAgICAgICBzZWFyY2hhYmxlPXtzZWFyY2hhYmxlfVxuICAgICAgICBzaXplPXtzaXplfVxuICAgICAgICB0ZXh0PXtcbiAgICAgICAgICA8ZGl2PlxuICAgICAgICAgICAge3NlYXJjaGFibGUgJiYgdHlwZW9mIGNoaWxkcmVuICE9PSAnZnVuY3Rpb24nID8gKFxuICAgICAgICAgICAgICA8U3R5bGVkU2VhcmNoSW5wdXRcbiAgICAgICAgICAgICAgICBzaXplPVwic21hbGxcIlxuICAgICAgICAgICAgICAgIG9uU2VhcmNoPXtvblNlYXJjaH1cbiAgICAgICAgICAgICAgICByZWY9e3NlYXJjaElucHV0UmVmfVxuICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgKSA6IG51bGx9XG4gICAgICAgICAgICA8TWVudUxpc3RcbiAgICAgICAgICAgICAgZGF0YS10ZXN0aWQ9e2RhdGFUZXN0SWR9XG4gICAgICAgICAgICAgIGNsYXNzTmFtZT17Y2xhc3NOYW1lfVxuICAgICAgICAgICAgICByb2xlPVwibWVudVwiXG4gICAgICAgICAgICA+XG4gICAgICAgICAgICAgIHtmaW5hbENoaWxkfVxuICAgICAgICAgICAgPC9NZW51TGlzdD5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgfVxuICAgICAgICBwb3J0YWxUYXJnZXQ9e3BvcnRhbFRhcmdldH1cbiAgICAgICAgZHluYW1pY0RvbVJlbmRlcmluZz17ZHluYW1pY0RvbVJlbmRlcmluZ31cbiAgICAgICAgYWxpZ249e2FsaWdufVxuICAgICAgPlxuICAgICAgICB7ZmluYWxEaXNjbG9zdXJlfVxuICAgICAgPC9TdHlsZWRQb3B1cD5cbiAgICApXG4gIH0sXG4pXG4iXX0= */"));
49
+ }) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/MenuContent.tsx"],"names":[],"mappings":"AAsE6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/MenuContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type { ButtonHTMLAttributes, MouseEvent, ReactNode, Ref } from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { SearchInput } from '../SearchInput'\nimport { Stack } from '../Stack'\nimport { useMenu } from './MenuProvider'\nimport { SIZES } from './constants'\nimport { searchChildren } from './helpers'\nimport type { MenuProps } from './types'\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size', 'searchable'].includes(prop),\n})<{ size?: keyof typeof SIZES; searchable: boolean }>`\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) => theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  min-width: ${SIZES.small};\n  max-width: ${SIZES.large};\n  ${({ size }) => (size ? `width: ${SIZES[size]};` : null)}\n  ${({ searchable }) => (searchable ? `min-width: 20rem` : null)};\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n`\n\nconst MenuList = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\nconst StyledSearchInput = styled(SearchInput)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nexport const Menu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget = document.body,\n      size,\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n      searchable = false,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const { isVisible, setIsVisible } = useMenu()\n    const searchInputRef = useRef<HTMLInputElement>(null)\n    const [localChild, setLocalChild] = useState<ReactNode[] | null>(null)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const disclosureRef = useRef<HTMLButtonElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      'aria-haspopup': 'dialog',\n      'aria-expanded': isVisible,\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    const onSearch = useCallback(\n      (value: string) => {\n        if (typeof children === 'object') {\n          setLocalChild(searchChildren(children, value))\n        }\n      },\n      [children],\n    )\n\n    useEffect(() => {\n      if (isVisible && searchable) {\n        setTimeout(() => {\n          searchInputRef.current?.focus()\n        }, 50)\n      }\n    }, [isVisible, searchable])\n\n    const finalChild = useMemo(() => {\n      if (typeof children === 'function') {\n        return children({ toggle: () => setIsVisible(!isVisible) })\n      }\n\n      if (searchable && localChild) {\n        return localChild\n      }\n\n      return children\n    }, [children, isVisible, localChild, searchable, setIsVisible])\n\n    return (\n      <StyledPopup\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        hideOnClickOutside\n        aria-label={ariaLabel}\n        className={className}\n        visible={triggerMethod === 'click' ? isVisible : undefined}\n        placement={placement}\n        hasArrow={hasArrow}\n        data-has-arrow={hasArrow}\n        role=\"dialog\"\n        id={finalId}\n        ref={popupRef}\n        onClose={() => {\n          setIsVisible(false)\n          setLocalChild(null)\n        }}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '30rem'}\n        maxWidth={maxWidth}\n        searchable={searchable}\n        size={size}\n        text={\n          <div>\n            {searchable && typeof children !== 'function' ? (\n              <StyledSearchInput\n                size=\"small\"\n                onSearch={onSearch}\n                ref={searchInputRef}\n              />\n            ) : null}\n            <MenuList\n              data-testid={dataTestId}\n              className={className}\n              role=\"menu\"\n            >\n              {finalChild}\n            </MenuList>\n          </div>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n        align={align}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n"]} */"));
50
50
  const Menu = forwardRef(({
51
51
  id,
52
52
  ariaLabel = "Menu",
@@ -59,7 +59,7 @@ const Menu = forwardRef(({
59
59
  maxHeight,
60
60
  maxWidth,
61
61
  portalTarget = document.body,
62
- size = "small",
62
+ size,
63
63
  triggerMethod = "click",
64
64
  dynamicDomRendering,
65
65
  align,
@@ -29,6 +29,9 @@ export type MenuProps = {
29
29
  * behavior by setting a portalTarget prop.
30
30
  */
31
31
  portalTarget?: HTMLElement;
32
+ /**
33
+ * @deprecated The size of the menu is automatic now to fit the content. Use this prop to set a fixed size.
34
+ */
32
35
  size?: keyof typeof SIZES;
33
36
  /**
34
37
  * The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.
@@ -18,7 +18,7 @@ const StyledBackdrop = /* @__PURE__ */ _styled__default.default("div", process.e
18
18
  theme
19
19
  }) => theme.colors.overlay, ";z-index:1;opacity:0;&[data-open='true']{padding:", ({
20
20
  theme
21
- }) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}&[data-visible='true']{opacity:1;}&[data-animation='true']{overflow:hidden;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx"],"names":[],"mappings":"AAc2D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  opacity: 0;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n\n  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n  &[data-animation='true'] {\n    overflow: hidden;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].includes(prop),\n})<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}rem;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\n}\n\nexport const Dialog = ({\n  children,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n  const {\n    registerModal,\n    unregisterModal,\n    openedModals,\n    previsousOpenedModales,\n  } = useModal()\n\n  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\n\n    return () => {\n      unregisterModal(id)\n    }\n  }, [id, registerModal, unregisterModal])\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    document.body.appendChild(element)\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [])\n\n  // Save the reassignment of eventHandler in the useEffect below\n  useEffect(() => {\n    onCloseRef.current = onClose\n  }, [onClose])\n\n  // On open focus the modal\n  useEffect(() => {\n    dialogRef.current?.focus()\n  }, [])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n\n      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node)\n      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // We need to reverse the array as the last opened modal should be the first to be with normal size\n  // while the first opened modal should shrink\n  const realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\n\n  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      css={backdropCss}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\n    >\n      <StyledDialog\n        css={dialogCss}\n        onKeyUp={handleKeyUp}\n        onKeyDown={handleFocusTrap}\n        className={className}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n        data-placement={placement}\n        data-size={size}\n        open\n        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
21
+ }) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}&[data-visible='true']{opacity:1;}&[data-animation='true']{overflow:hidden;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx"],"names":[],"mappings":"AAc2D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  opacity: 0;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n\n  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n  &[data-animation='true'] {\n    overflow: hidden;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].includes(prop),\n})<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}rem;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\n}\n\nexport const Dialog = ({\n  children,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n  const {\n    registerModal,\n    unregisterModal,\n    openedModals,\n    previsousOpenedModales,\n  } = useModal()\n\n  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\n\n    return () => {\n      unregisterModal(id)\n    }\n  }, [id, registerModal, unregisterModal])\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    document.body.appendChild(element)\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [])\n\n  // Save the reassignment of eventHandler in the useEffect below\n  useEffect(() => {\n    onCloseRef.current = onClose\n  }, [onClose])\n\n  // On open focus the modal\n  useEffect(() => {\n    dialogRef.current?.focus()\n  }, [])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // We need to reverse the array as the last opened modal should be the first to be with normal size\n  // while the first opened modal should shrink\n  const realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node) &&\n        position === 0\n      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside, position],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      css={backdropCss}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\n    >\n      <StyledDialog\n        css={dialogCss}\n        onKeyUp={handleKeyUp}\n        onKeyDown={handleFocusTrap}\n        className={className}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n        data-placement={placement}\n        data-size={size}\n        open\n        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
22
22
  const StyledDialog = /* @__PURE__ */ _styled__default.default("dialog", process.env.NODE_ENV === "production" ? {
23
23
  shouldForwardProp: (prop) => !["position", "size", "openedModals", "top"].includes(prop),
24
24
  target: "e1fims040"
@@ -49,7 +49,7 @@ const StyledDialog = /* @__PURE__ */ _styled__default.default("dialog", process.
49
49
  }) => position > 0 ? `
50
50
  width: calc(${constants.MODAL_WIDTH[size]}rem - ${position * 50}px) !important;
51
51
  transform: translate3d(0, -${top}px, 0);
52
- ` : void 0, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx"],"names":[],"mappings":"AAuDqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  opacity: 0;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n\n  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n  &[data-animation='true'] {\n    overflow: hidden;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].includes(prop),\n})<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}rem;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\n}\n\nexport const Dialog = ({\n  children,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n  const {\n    registerModal,\n    unregisterModal,\n    openedModals,\n    previsousOpenedModales,\n  } = useModal()\n\n  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\n\n    return () => {\n      unregisterModal(id)\n    }\n  }, [id, registerModal, unregisterModal])\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    document.body.appendChild(element)\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [])\n\n  // Save the reassignment of eventHandler in the useEffect below\n  useEffect(() => {\n    onCloseRef.current = onClose\n  }, [onClose])\n\n  // On open focus the modal\n  useEffect(() => {\n    dialogRef.current?.focus()\n  }, [])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n\n      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node)\n      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // We need to reverse the array as the last opened modal should be the first to be with normal size\n  // while the first opened modal should shrink\n  const realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\n\n  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      css={backdropCss}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\n    >\n      <StyledDialog\n        css={dialogCss}\n        onKeyUp={handleKeyUp}\n        onKeyDown={handleFocusTrap}\n        className={className}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n        data-placement={placement}\n        data-size={size}\n        open\n        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
52
+ ` : void 0, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx"],"names":[],"mappings":"AAuDqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  opacity: 0;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n\n  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n  &[data-animation='true'] {\n    overflow: hidden;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].includes(prop),\n})<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}rem;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\n}\n\nexport const Dialog = ({\n  children,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n  const {\n    registerModal,\n    unregisterModal,\n    openedModals,\n    previsousOpenedModales,\n  } = useModal()\n\n  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\n\n    return () => {\n      unregisterModal(id)\n    }\n  }, [id, registerModal, unregisterModal])\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    document.body.appendChild(element)\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [])\n\n  // Save the reassignment of eventHandler in the useEffect below\n  useEffect(() => {\n    onCloseRef.current = onClose\n  }, [onClose])\n\n  // On open focus the modal\n  useEffect(() => {\n    dialogRef.current?.focus()\n  }, [])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // We need to reverse the array as the last opened modal should be the first to be with normal size\n  // while the first opened modal should shrink\n  const realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node) &&\n        position === 0\n      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside, position],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      css={backdropCss}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\n    >\n      <StyledDialog\n        css={dialogCss}\n        onKeyUp={handleKeyUp}\n        onKeyDown={handleFocusTrap}\n        className={className}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n        data-placement={placement}\n        data-size={size}\n        open\n        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
53
53
  const stopCancel = (event) => {
54
54
  event.preventDefault();
55
55
  event.stopPropagation();
@@ -119,6 +119,11 @@ const Dialog = ({
119
119
  const stopFocus = React.useCallback((event) => {
120
120
  event.stopPropagation();
121
121
  }, []);
122
+ const realPosition = [...openedModals].findIndex((object) => object.id === id);
123
+ const position = [...openedModals].reverse().findIndex((object) => object.id === id);
124
+ const modalAbove = openedModals[realPosition + 1];
125
+ const currentModalHeight = dialogRef.current?.offsetHeight;
126
+ let top = 0;
122
127
  const handleKeyUp = React.useCallback((event) => {
123
128
  event.stopPropagation();
124
129
  if (event.key === "Escape" && hideOnEsc) {
@@ -127,11 +132,10 @@ const Dialog = ({
127
132
  }
128
133
  }, [hideOnEsc]);
129
134
  const handleClose = React.useCallback((event) => {
130
- event.stopPropagation();
131
- if (hideOnClickOutside && dialogRef.current && !dialogRef.current.contains(event.target)) {
135
+ if (hideOnClickOutside && dialogRef.current && !dialogRef.current.contains(event.target) && position === 0) {
132
136
  onCloseRef.current();
133
137
  }
134
- }, [hideOnClickOutside]);
138
+ }, [hideOnClickOutside, position]);
135
139
  const handleFocusTrap = React.useCallback((event) => {
136
140
  event.stopPropagation();
137
141
  if (event.key === "Escape") {
@@ -158,11 +162,6 @@ const Dialog = ({
158
162
  event.preventDefault();
159
163
  }
160
164
  }, []);
161
- const realPosition = [...openedModals].findIndex((object) => object.id === id);
162
- const position = [...openedModals].reverse().findIndex((object) => object.id === id);
163
- const modalAbove = openedModals[realPosition + 1];
164
- const currentModalHeight = dialogRef.current?.offsetHeight;
165
- let top = 0;
166
165
  if (modalAbove?.ref && typeof modalAbove.ref === "object" && "current" in modalAbove.ref && currentModalHeight) {
167
166
  top = (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 - currentModalHeight / 2 + 20;
168
167
  }