@ultraviolet/ui 1.59.1 → 1.61.0
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.
- package/dist/components/ActionBar/index.cjs +2 -2
- package/dist/components/ActionBar/index.js +2 -2
- package/dist/components/DateInput/index.cjs +6 -4
- package/dist/components/DateInput/index.js +6 -4
- package/dist/components/Expandable/index.cjs +36 -21
- package/dist/components/Expandable/index.d.ts +3 -0
- package/dist/components/Expandable/index.js +36 -21
- package/dist/components/MenuV2/index.cjs +6 -6
- package/dist/components/MenuV2/index.js +6 -6
- package/dist/components/Modal/Dialog.cjs +4 -4
- package/dist/components/Modal/Dialog.js +4 -4
- package/dist/components/Notification/index.cjs +1 -1
- package/dist/components/Notification/index.js +1 -1
- package/dist/components/SearchInput/index.cjs +24 -4
- package/dist/components/SearchInput/index.d.ts +2 -32
- package/dist/components/SearchInput/index.js +24 -4
- package/dist/components/SearchInput/types.d.ts +1 -1
- package/dist/components/SelectInputV2/Dropdown.cjs +12 -12
- package/dist/components/SelectInputV2/Dropdown.js +12 -12
- package/package.json +3 -3
|
@@ -9,11 +9,11 @@ const StyledExpandable = /* @__PURE__ */ _styled("div", process.env.NODE_ENV ===
|
|
|
9
9
|
shouldForwardProp: (prop) => !["animationDuration"].includes(prop),
|
|
10
10
|
target: "e5hc7t70",
|
|
11
11
|
label: "StyledExpandable"
|
|
12
|
-
})("transition:max-height
|
|
12
|
+
})('&[data-is-animated="true"]{transition:max-height ', ({
|
|
13
13
|
animationDuration
|
|
14
14
|
}) => animationDuration, "ms ease-out,opacity ", ({
|
|
15
15
|
animationDuration
|
|
16
|
-
}) => animationDuration, "ms ease-out;height:auto;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
16
|
+
}) => animationDuration, "ms ease-out;}height:auto;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL0V4cGFuZGFibGUvaW5kZXgudHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQTZCaUMiLCJmaWxlIjoiL2hvbWUvcnVubmVyL3dvcmsvdWx0cmF2aW9sZXQvdWx0cmF2aW9sZXQvcGFja2FnZXMvdWkvc3JjL2NvbXBvbmVudHMvRXhwYW5kYWJsZS9pbmRleC50c3giLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCdcbmltcG9ydCB0eXBlIHsgUmVhY3ROb2RlIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyB1c2VFZmZlY3QsIHVzZVJlZiwgdXNlU3RhdGUgfSBmcm9tICdyZWFjdCdcblxuY29uc3QgQU5JTUFUSU9OX0RVUkFUSU9OID0gMzAwIC8vIGluIG1zXG5cbnR5cGUgRXhwYW5kYWJsZVByb3BzID0ge1xuICAvKipcbiAgICogVGhlIGNvbnRlbnQgdG8gZGlzcGxheVxuICAgKi9cbiAgY2hpbGRyZW46IFJlYWN0Tm9kZVxuICAvKipcbiAgICogVG8gZGlzcGxheSBvciBub3QgdGhlIGNvbnRlbnRcbiAgICovXG4gIG9wZW5lZD86IGJvb2xlYW5cbiAgLyoqXG4gICAqIFRoZSBtaW5pbXVtIGhlaWdodCBvZiB0aGUgY29udGVudFxuICAgKi9cbiAgbWluSGVpZ2h0PzogbnVtYmVyXG4gIGNsYXNzTmFtZT86IHN0cmluZ1xuICAnZGF0YS10ZXN0aWQnPzogc3RyaW5nXG4gIC8qKlxuICAgKiBUaGUgZHVyYXRpb24gb2YgdGhlIGFuaW1hdGlvbiBpbiBtcy4gSWYgc2V0IHRvIDAsIHRoZSBhbmltYXRpb24gd2lsbCBiZSBkaXNhYmxlZC5cbiAgICovXG4gIGFuaW1hdGlvbkR1cmF0aW9uPzogbnVtYmVyXG59XG5cbmV4cG9ydCBjb25zdCBTdHlsZWRFeHBhbmRhYmxlID0gc3R5bGVkKCdkaXYnLCB7XG4gIHNob3VsZEZvcndhcmRQcm9wOiBwcm9wID0+ICFbJ2FuaW1hdGlvbkR1cmF0aW9uJ10uaW5jbHVkZXMocHJvcCksXG59KTx7IGFuaW1hdGlvbkR1cmF0aW9uOiBudW1iZXIgfT5gXG4gICZbZGF0YS1pcy1hbmltYXRlZD1cInRydWVcIl0ge1xuICAgIHRyYW5zaXRpb246XG4gICAgICBtYXgtaGVpZ2h0ICR7KHsgYW5pbWF0aW9uRHVyYXRpb24gfSkgPT4gYW5pbWF0aW9uRHVyYXRpb259bXMgZWFzZS1vdXQsXG4gICAgICBvcGFjaXR5ICR7KHsgYW5pbWF0aW9uRHVyYXRpb24gfSkgPT4gYW5pbWF0aW9uRHVyYXRpb259bXMgZWFzZS1vdXQ7XG4gICAgfVxuICBoZWlnaHQ6IGF1dG87XG5gXG5cbi8qKlxuICogVGhlIEV4cGFuZGFibGUgY29tcG9uZW50IGlzIGEgZHluYW1pYyBSZWFjdCBjb21wb25lbnQgdGhhdCBhbGxvd3MgZm9yIHRoZSBleHBhbnNpb24gb2YgaXRzIGNoaWxkcmVuIGNvbnRlbnRcbiAqIGJhc2VkIG9uIGl0cyBoZWlnaHQuIFRoZSBjb21wb25lbnQgY29tZXMgd2l0aCBhIHNsZWVrIGFuZCBzbW9vdGggYW5pbWF0aW9uLCBwcm92aWRpbmcgYSB2aXN1YWxseSBwbGVhc2luZ1xuICogdXNlciBleHBlcmllbmNlLlxuICovXG5leHBvcnQgY29uc3QgRXhwYW5kYWJsZSA9ICh7XG4gIGNoaWxkcmVuLFxuICBvcGVuZWQsXG4gIG1pbkhlaWdodCA9IDAsXG4gIGNsYXNzTmFtZSxcbiAgJ2RhdGEtdGVzdGlkJzogZGF0YVRlc3RJZCxcbiAgYW5pbWF0aW9uRHVyYXRpb24gPSBBTklNQVRJT05fRFVSQVRJT04sXG59OiBFeHBhbmRhYmxlUHJvcHMpID0+IHtcbiAgY29uc3QgW2hlaWdodCwgc2V0SGVpZ2h0XSA9IHVzZVN0YXRlPG51bWJlciB8IG51bGw+KG51bGwpXG4gIGNvbnN0IHRyYW5zaXRpb25UaW1lciA9IHVzZVJlZjxSZXR1cm5UeXBlPHR5cGVvZiBzZXRUaW1lb3V0PiB8IHVuZGVmaW5lZD4oKVxuICBjb25zdCByZWYgPSB1c2VSZWY8SFRNTERpdkVsZW1lbnQ+KG51bGwpXG4gIGNvbnN0IHNob3VsZEJlQW5pbWF0ZWQgPSBhbmltYXRpb25EdXJhdGlvbiA+IDBcblxuICAvKipcbiAgICogQXQgbW91bnQsIHdlIHNldCB0aGUgaGVpZ2h0IHZhcmlhYmxlIHRvIHRoZSBoZWlnaHQgb2YgdGhlIGNvbnRlbnQgb25seSBpZiB0aGUgY29tcG9uZW50IGlzIGNsb3NlZC5cbiAgICogVGhpcyBpcyB0byBlbnN1cmUgd2UgZG9uJ3QgaGF2ZSBhbmltYXRpb24gd2hlbiB0aGUgY29tcG9uZW50IGlzIG9wZW5lZCBhdCBtb3VudC5cbiAgICovXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKHJlZi5jdXJyZW50KSB7XG4gICAgICBzZXRIZWlnaHQocmVmLmN1cnJlbnQuc2Nyb2xsSGVpZ2h0ID8/IDApXG4gICAgfVxuICB9LCBbcmVmLmN1cnJlbnQ/LnNjcm9sbEhlaWdodF0pXG5cbiAgLyoqXG4gICAqIEhlcmUgd2Ugc2V0IG1heEhlaWdodCB0byB0aGUgaGVpZ2h0IG9mIHRoZSBjb250ZW50IHdoZW4gdGhlIGNvbXBvbmVudCBpcyBvcGVuZWRcbiAgICogYW5kIGFmdGVyIDMwMG1zIHdlIHNldCBtYXhIZWlnaHQgdG8gaW5pdGlhbCB0byBsZXQgdGhlIGNvbnRlbnQgZ3JvdyB3aXRoIGFuaW1hdGlvbi5cbiAgICogU2V0dGluZyBpdCB0byBpbml0aWFsIGlzIHJlcXVpcmVkIHRvIGJlIGFibGUgdG8gaGF2ZSBuZXN0ZWQgZXhwYW5kYWJsZSBvciB0aGUgaGVpZ2h0IHdvbid0IGZvbGxvdy5cbiAgICovXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKG9wZW5lZCAmJiByZWYuY3VycmVudCAmJiBoZWlnaHQpIHtcbiAgICAgIHJlZi5jdXJyZW50LnN0eWxlLm1heEhlaWdodCA9IGAke2hlaWdodH1weGBcbiAgICAgIHJlZi5jdXJyZW50LnN0eWxlLnZpc2liaWxpdHkgPSAnJ1xuICAgICAgaWYgKHNob3VsZEJlQW5pbWF0ZWQpIHtcbiAgICAgICAgdHJhbnNpdGlvblRpbWVyLmN1cnJlbnQgPSBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgICAgICBpZiAocmVmLmN1cnJlbnQpIHtcbiAgICAgICAgICAgIHJlZi5jdXJyZW50LnN0eWxlLm1heEhlaWdodCA9ICdpbml0aWFsJ1xuICAgICAgICAgICAgcmVmLmN1cnJlbnQuc3R5bGUub3ZlcmZsb3cgPSAndmlzaWJsZSdcbiAgICAgICAgICAgIHJlZi5jdXJyZW50LnN0eWxlLnZpc2liaWxpdHkgPSAnJ1xuICAgICAgICAgIH1cbiAgICAgICAgfSwgQU5JTUFUSU9OX0RVUkFUSU9OKVxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgcmVmLmN1cnJlbnQuc3R5bGUubWF4SGVpZ2h0ID0gJ2luaXRpYWwnXG4gICAgICAgIHJlZi5jdXJyZW50LnN0eWxlLm92ZXJmbG93ID0gJ3Zpc2libGUnXG4gICAgICAgIHJlZi5jdXJyZW50LnN0eWxlLnZpc2liaWxpdHkgPSAnJ1xuICAgICAgfVxuICAgIH0gZWxzZSB7XG4gICAgICBjbGVhclRpbWVvdXQodHJhbnNpdGlvblRpbWVyLmN1cnJlbnQpXG5cbiAgICAgIGlmIChyZWYuY3VycmVudCAmJiBoZWlnaHQpIHtcbiAgICAgICAgcmVmLmN1cnJlbnQuc3R5bGUubWF4SGVpZ2h0ID0gYCR7aGVpZ2h0fXB4YFxuICAgICAgICBpZiAoc2hvdWxkQmVBbmltYXRlZCkge1xuICAgICAgICAgIHRyYW5zaXRpb25UaW1lci5jdXJyZW50ID0gc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICAgICAgICBpZiAocmVmLmN1cnJlbnQpIHtcbiAgICAgICAgICAgICAgcmVmLmN1cnJlbnQuc3R5bGUubWF4SGVpZ2h0ID0gYCR7bWluSGVpZ2h0fXB4YFxuICAgICAgICAgICAgICByZWYuY3VycmVudC5zdHlsZS5vdmVyZmxvdyA9ICdoaWRkZW4nXG4gICAgICAgICAgICAgIHNldFRpbWVvdXQoKCkgPT4ge1xuICAgICAgICAgICAgICAgIGlmIChyZWYuY3VycmVudCAmJiAhbWluSGVpZ2h0KSB7XG4gICAgICAgICAgICAgICAgICByZWYuY3VycmVudC5zdHlsZS52aXNpYmlsaXR5ID0gJ2hpZGRlbidcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIH0sIEFOSU1BVElPTl9EVVJBVElPTilcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9LCAwKVxuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIHJlZi5jdXJyZW50LnN0eWxlLm1heEhlaWdodCA9IGAke21pbkhlaWdodH1weGBcbiAgICAgICAgICByZWYuY3VycmVudC5zdHlsZS5vdmVyZmxvdyA9ICdoaWRkZW4nXG4gICAgICAgICAgaWYgKCFtaW5IZWlnaHQpIHtcbiAgICAgICAgICAgIHJlZi5jdXJyZW50LnN0eWxlLnZpc2liaWxpdHkgPSAnaGlkZGVuJ1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cblxuICAgIHJldHVybiAoKSA9PiB7XG4gICAgICBjbGVhclRpbWVvdXQodHJhbnNpdGlvblRpbWVyLmN1cnJlbnQpXG4gICAgfVxuICB9LCBbYW5pbWF0aW9uRHVyYXRpb24sIGhlaWdodCwgbWluSGVpZ2h0LCBvcGVuZWQsIHNob3VsZEJlQW5pbWF0ZWRdKVxuXG4gIHJldHVybiAoXG4gICAgPFN0eWxlZEV4cGFuZGFibGVcbiAgICAgIGRhdGEtdGVzdGlkPXtkYXRhVGVzdElkfVxuICAgICAgcmVmPXtyZWZ9XG4gICAgICBjbGFzc05hbWU9e2NsYXNzTmFtZX1cbiAgICAgIGFuaW1hdGlvbkR1cmF0aW9uPXthbmltYXRpb25EdXJhdGlvbn1cbiAgICAgIGRhdGEtaXMtYW5pbWF0ZWQ9e3Nob3VsZEJlQW5pbWF0ZWR9XG4gICAgPlxuICAgICAge2NoaWxkcmVufVxuICAgIDwvU3R5bGVkRXhwYW5kYWJsZT5cbiAgKVxufVxuIl19 */"));
|
|
17
17
|
const Expandable = ({
|
|
18
18
|
children,
|
|
19
19
|
opened,
|
|
@@ -25,6 +25,7 @@ const Expandable = ({
|
|
|
25
25
|
const [height, setHeight] = useState(null);
|
|
26
26
|
const transitionTimer = useRef();
|
|
27
27
|
const ref = useRef(null);
|
|
28
|
+
const shouldBeAnimated = animationDuration > 0;
|
|
28
29
|
useEffect(() => {
|
|
29
30
|
if (ref.current) {
|
|
30
31
|
setHeight(ref.current.scrollHeight ?? 0);
|
|
@@ -34,35 +35,49 @@ const Expandable = ({
|
|
|
34
35
|
if (opened && ref.current && height) {
|
|
35
36
|
ref.current.style.maxHeight = `${height}px`;
|
|
36
37
|
ref.current.style.visibility = "";
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
ref.current
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
38
|
+
if (shouldBeAnimated) {
|
|
39
|
+
transitionTimer.current = setTimeout(() => {
|
|
40
|
+
if (ref.current) {
|
|
41
|
+
ref.current.style.maxHeight = "initial";
|
|
42
|
+
ref.current.style.overflow = "visible";
|
|
43
|
+
ref.current.style.visibility = "";
|
|
44
|
+
}
|
|
45
|
+
}, ANIMATION_DURATION);
|
|
46
|
+
} else {
|
|
47
|
+
ref.current.style.maxHeight = "initial";
|
|
48
|
+
ref.current.style.overflow = "visible";
|
|
49
|
+
ref.current.style.visibility = "";
|
|
50
|
+
}
|
|
44
51
|
} else {
|
|
45
52
|
clearTimeout(transitionTimer.current);
|
|
46
53
|
if (ref.current && height) {
|
|
47
54
|
ref.current.style.maxHeight = `${height}px`;
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
ref.current
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
ref.current
|
|
55
|
-
|
|
56
|
-
|
|
55
|
+
if (shouldBeAnimated) {
|
|
56
|
+
transitionTimer.current = setTimeout(() => {
|
|
57
|
+
if (ref.current) {
|
|
58
|
+
ref.current.style.maxHeight = `${minHeight}px`;
|
|
59
|
+
ref.current.style.overflow = "hidden";
|
|
60
|
+
setTimeout(() => {
|
|
61
|
+
if (ref.current && !minHeight) {
|
|
62
|
+
ref.current.style.visibility = "hidden";
|
|
63
|
+
}
|
|
64
|
+
}, ANIMATION_DURATION);
|
|
65
|
+
}
|
|
66
|
+
}, 0);
|
|
67
|
+
} else {
|
|
68
|
+
ref.current.style.maxHeight = `${minHeight}px`;
|
|
69
|
+
ref.current.style.overflow = "hidden";
|
|
70
|
+
if (!minHeight) {
|
|
71
|
+
ref.current.style.visibility = "hidden";
|
|
57
72
|
}
|
|
58
|
-
}
|
|
73
|
+
}
|
|
59
74
|
}
|
|
60
75
|
}
|
|
61
76
|
return () => {
|
|
62
77
|
clearTimeout(transitionTimer.current);
|
|
63
78
|
};
|
|
64
|
-
}, [height, minHeight, opened]);
|
|
65
|
-
return /* @__PURE__ */ jsx(StyledExpandable, { "data-testid": dataTestId, ref, className, animationDuration, children });
|
|
79
|
+
}, [animationDuration, height, minHeight, opened, shouldBeAnimated]);
|
|
80
|
+
return /* @__PURE__ */ jsx(StyledExpandable, { "data-testid": dataTestId, ref, className, animationDuration, "data-is-animated": shouldBeAnimated, children });
|
|
66
81
|
};
|
|
67
82
|
export {
|
|
68
83
|
Expandable,
|
|
@@ -23,15 +23,15 @@ const StyledPopup = /* @__PURE__ */ _styled__default.default(index.Popup, proces
|
|
|
23
23
|
label: "StyledPopup"
|
|
24
24
|
})("background-color:", ({
|
|
25
25
|
theme
|
|
26
|
-
}) => theme.colors.
|
|
26
|
+
}) => theme.colors.other.elevation.background.raised, ";box-shadow:", ({
|
|
27
27
|
theme
|
|
28
|
-
}) => theme.shadows.
|
|
28
|
+
}) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`, ";padding:0;&[data-has-arrow='true']{&::after{border-color:", ({
|
|
29
29
|
theme
|
|
30
|
-
}) => theme.colors.
|
|
30
|
+
}) => theme.colors.other.elevation.background.raised, " transparent transparent transparent;}}width:", ({
|
|
31
31
|
size
|
|
32
32
|
}) => SIZES[size], ";max-width:none;padding:", ({
|
|
33
33
|
theme
|
|
34
|
-
}) => `${theme.space["0.25"]} 0`, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01lbnVWMi9pbmRleC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBeUNnQyIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9NZW51VjIvaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7XG4gIEJ1dHRvbkhUTUxBdHRyaWJ1dGVzLFxuICBDb21wb25lbnRQcm9wcyxcbiAgTW91c2VFdmVudCxcbiAgUmVhY3RFbGVtZW50LFxuICBSZWFjdE5vZGUsXG4gIFJlZixcbn0gZnJvbSAncmVhY3QnXG5pbXBvcnQge1xuICBjbG9uZUVsZW1lbnQsXG4gIGZvcndhcmRSZWYsXG4gIGlzVmFsaWRFbGVtZW50LFxuICB1c2VJZCxcbiAgdXNlSW1wZXJhdGl2ZUhhbmRsZSxcbiAgdXNlUmVmLFxuICB1c2VTdGF0ZSxcbn0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBQb3B1cCB9IGZyb20gJy4uL1BvcHVwJ1xuaW1wb3J0IHsgU3RhY2sgfSBmcm9tICcuLi9TdGFjaydcbmltcG9ydCB7IEdyb3VwIH0gZnJvbSAnLi9Hcm91cCdcbmltcG9ydCBJdGVtIGZyb20gJy4vSXRlbSdcblxuY29uc3QgU0laRVMgPSB7XG4gIHNtYWxsOiAnMTgwcHgnLFxuICBtZWRpdW06ICcyODBweCcsXG4gIGxhcmdlOiAnMzgwcHgnLFxufVxuXG5leHBvcnQgdHlwZSBEaXNjbG9zdXJlUHJvcHMgPSB7IHZpc2libGU6IGJvb2xlYW4gfVxuXG50eXBlIERpc2Nsb3N1cmVFbGVtZW50ID1cbiAgfCAoKFxuICAgICAgZGlzY2xvc3VyZTogRGlzY2xvc3VyZVByb3BzLFxuICAgICkgPT4gUmVhY3RFbGVtZW50PEJ1dHRvbkhUTUxBdHRyaWJ1dGVzPEhUTUxCdXR0b25FbGVtZW50Pj4pXG4gIHwgKFJlYWN0RWxlbWVudDxCdXR0b25IVE1MQXR0cmlidXRlczxIVE1MQnV0dG9uRWxlbWVudD4+ICYge1xuICAgICAgcmVmPzogUmVmPEhUTUxCdXR0b25FbGVtZW50PlxuICAgIH0pXG5cbmNvbnN0IFN0eWxlZFBvcHVwID0gc3R5bGVkKFBvcHVwLCB7XG4gIHNob3VsZEZvcndhcmRQcm9wOiBwcm9wID0+ICFbJ3NpemUnXS5pbmNsdWRlcyhwcm9wKSxcbn0pPHsgc2l6ZToga2V5b2YgdHlwZW9mIFNJWkVTIH0+YFxuICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+
|
|
34
|
+
}) => `${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/index.tsx"],"names":[],"mappings":"AAyCgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  ComponentProps,\n  MouseEvent,\n  ReactElement,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useId,\n  useImperativeHandle,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Group } from './Group'\nimport Item from './Item'\n\nconst SIZES = {\n  small: '180px',\n  medium: '280px',\n  large: '380px',\n}\n\nexport type DisclosureProps = { visible: boolean }\n\ntype DisclosureElement =\n  | ((\n      disclosure: DisclosureProps,\n    ) => ReactElement<ButtonHTMLAttributes<HTMLButtonElement>>)\n  | (ReactElement<ButtonHTMLAttributes<HTMLButtonElement>> & {\n      ref?: Ref<HTMLButtonElement>\n    })\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: keyof typeof SIZES }>`\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  width: ${({ size }) => SIZES[size]};\n  max-width: none;\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\ntype ChildMenuProps = {\n  toggle: () => void\n}\n\ntype MenuProps = {\n  id?: string\n  ariaLabel?: string\n  children?: ReactNode | (({ toggle }: ChildMenuProps) => ReactNode)\n  className?: string\n  disclosure: DisclosureElement\n  hasArrow?: boolean\n  visible?: boolean\n  'data-testid'?: string\n  maxHeight?: string\n  /**\n   * @deprecated: use `size` instead\n   */\n  maxWidth?: string\n  /**\n   * By default, the portal target is children container or document.body if children is a function. You can override this\n   * behavior by setting a portalTarget prop.\n   */\n  portalTarget?: HTMLElement\n  size?: keyof typeof SIZES\n  /**\n   * The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.\n   * If set to `hover`, the menu will open when the user hovers over the disclosure.\n   */\n  triggerMethod?: 'click' | 'hover'\n} & Pick<ComponentProps<typeof Popup>, 'placement' | 'dynamicDomRendering'>\n\nconst FwdMenu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      visible = false,\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget,\n      size = 'small',\n      triggerMethod = 'click',\n      dynamicDomRendering,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const [isVisible, setIsVisible] = useState(visible)\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    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={() => setIsVisible(false)}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '480px'}\n        maxWidth={maxWidth}\n        size={size}\n        text={\n          <MenuList data-testid={dataTestId} className={className} role=\"menu\">\n            {typeof children === 'function'\n              ? children({ toggle: () => setIsVisible(!isVisible) })\n              : children}\n          </MenuList>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n\n/**\n * A menu is a widget that offers a list of choices to the user, such as a set of actions or functions.\n * A menu is usually opened, or made visible, by activating a menu button, choosing an item in a menu that opens a\n * sub menu, or by invoking a command, such as `Shift + F10` on Windows, that opens a context specific menu.\n * When a user activates a choice in a menu, the menu usually closes unless the choice opened a submenu.\n */\nexport const MenuV2 = Object.assign(FwdMenu, { Item, Group })\n"]} */"));
|
|
35
35
|
const MenuList = /* @__PURE__ */ _styled__default.default(index$1.Stack, process.env.NODE_ENV === "production" ? {
|
|
36
36
|
target: "e1jn11gg0"
|
|
37
37
|
} : {
|
|
@@ -39,11 +39,11 @@ const MenuList = /* @__PURE__ */ _styled__default.default(index$1.Stack, process
|
|
|
39
39
|
label: "MenuList"
|
|
40
40
|
})("overflow-y:auto;overflow-x:hidden;&:after,&:before{border:solid transparent;border-width:9px;content:' ';height:0;width:0;position:absolute;pointer-events:none;}&:after{border-color:transparent;}&:before{border-color:transparent;}background-color:", ({
|
|
41
41
|
theme
|
|
42
|
-
}) => theme.colors.
|
|
42
|
+
}) => theme.colors.other.elevation.background.raised, ";color:", ({
|
|
43
43
|
theme
|
|
44
44
|
}) => theme.colors.neutral.text, ";border-radius:", ({
|
|
45
45
|
theme
|
|
46
|
-
}) => theme.radii.default, ";position:relative;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01lbnVWMi9pbmRleC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBMEQ4QiIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9NZW51VjIvaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7XG4gIEJ1dHRvbkhUTUxBdHRyaWJ1dGVzLFxuICBDb21wb25lbnRQcm9wcyxcbiAgTW91c2VFdmVudCxcbiAgUmVhY3RFbGVtZW50LFxuICBSZWFjdE5vZGUsXG4gIFJlZixcbn0gZnJvbSAncmVhY3QnXG5pbXBvcnQge1xuICBjbG9uZUVsZW1lbnQsXG4gIGZvcndhcmRSZWYsXG4gIGlzVmFsaWRFbGVtZW50LFxuICB1c2VJZCxcbiAgdXNlSW1wZXJhdGl2ZUhhbmRsZSxcbiAgdXNlUmVmLFxuICB1c2VTdGF0ZSxcbn0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBQb3B1cCB9IGZyb20gJy4uL1BvcHVwJ1xuaW1wb3J0IHsgU3RhY2sgfSBmcm9tICcuLi9TdGFjaydcbmltcG9ydCB7IEdyb3VwIH0gZnJvbSAnLi9Hcm91cCdcbmltcG9ydCBJdGVtIGZyb20gJy4vSXRlbSdcblxuY29uc3QgU0laRVMgPSB7XG4gIHNtYWxsOiAnMTgwcHgnLFxuICBtZWRpdW06ICcyODBweCcsXG4gIGxhcmdlOiAnMzgwcHgnLFxufVxuXG5leHBvcnQgdHlwZSBEaXNjbG9zdXJlUHJvcHMgPSB7IHZpc2libGU6IGJvb2xlYW4gfVxuXG50eXBlIERpc2Nsb3N1cmVFbGVtZW50ID1cbiAgfCAoKFxuICAgICAgZGlzY2xvc3VyZTogRGlzY2xvc3VyZVByb3BzLFxuICAgICkgPT4gUmVhY3RFbGVtZW50PEJ1dHRvbkhUTUxBdHRyaWJ1dGVzPEhUTUxCdXR0b25FbGVtZW50Pj4pXG4gIHwgKFJlYWN0RWxlbWVudDxCdXR0b25IVE1MQXR0cmlidXRlczxIVE1MQnV0dG9uRWxlbWVudD4+ICYge1xuICAgICAgcmVmPzogUmVmPEhUTUxCdXR0b25FbGVtZW50PlxuICAgIH0pXG5cbmNvbnN0IFN0eWxlZFBvcHVwID0gc3R5bGVkKFBvcHVwLCB7XG4gIHNob3VsZEZvcndhcmRQcm9wOiBwcm9wID0+ICFbJ3NpemUnXS5pbmNsdWRlcyhwcm9wKSxcbn0pPHsgc2l6ZToga2V5b2YgdHlwZW9mIFNJWkVTIH0+YFxuICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+
|
|
46
|
+
}) => 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/index.tsx"],"names":[],"mappings":"AA0D8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  ComponentProps,\n  MouseEvent,\n  ReactElement,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useId,\n  useImperativeHandle,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Group } from './Group'\nimport Item from './Item'\n\nconst SIZES = {\n  small: '180px',\n  medium: '280px',\n  large: '380px',\n}\n\nexport type DisclosureProps = { visible: boolean }\n\ntype DisclosureElement =\n  | ((\n      disclosure: DisclosureProps,\n    ) => ReactElement<ButtonHTMLAttributes<HTMLButtonElement>>)\n  | (ReactElement<ButtonHTMLAttributes<HTMLButtonElement>> & {\n      ref?: Ref<HTMLButtonElement>\n    })\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: keyof typeof SIZES }>`\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  width: ${({ size }) => SIZES[size]};\n  max-width: none;\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\ntype ChildMenuProps = {\n  toggle: () => void\n}\n\ntype MenuProps = {\n  id?: string\n  ariaLabel?: string\n  children?: ReactNode | (({ toggle }: ChildMenuProps) => ReactNode)\n  className?: string\n  disclosure: DisclosureElement\n  hasArrow?: boolean\n  visible?: boolean\n  'data-testid'?: string\n  maxHeight?: string\n  /**\n   * @deprecated: use `size` instead\n   */\n  maxWidth?: string\n  /**\n   * By default, the portal target is children container or document.body if children is a function. You can override this\n   * behavior by setting a portalTarget prop.\n   */\n  portalTarget?: HTMLElement\n  size?: keyof typeof SIZES\n  /**\n   * The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.\n   * If set to `hover`, the menu will open when the user hovers over the disclosure.\n   */\n  triggerMethod?: 'click' | 'hover'\n} & Pick<ComponentProps<typeof Popup>, 'placement' | 'dynamicDomRendering'>\n\nconst FwdMenu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      visible = false,\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget,\n      size = 'small',\n      triggerMethod = 'click',\n      dynamicDomRendering,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const [isVisible, setIsVisible] = useState(visible)\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    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={() => setIsVisible(false)}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '480px'}\n        maxWidth={maxWidth}\n        size={size}\n        text={\n          <MenuList data-testid={dataTestId} className={className} role=\"menu\">\n            {typeof children === 'function'\n              ? children({ toggle: () => setIsVisible(!isVisible) })\n              : children}\n          </MenuList>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n\n/**\n * A menu is a widget that offers a list of choices to the user, such as a set of actions or functions.\n * A menu is usually opened, or made visible, by activating a menu button, choosing an item in a menu that opens a\n * sub menu, or by invoking a command, such as `Shift + F10` on Windows, that opens a context specific menu.\n * When a user activates a choice in a menu, the menu usually closes unless the choice opened a submenu.\n */\nexport const MenuV2 = Object.assign(FwdMenu, { Item, Group })\n"]} */"));
|
|
47
47
|
const FwdMenu = React.forwardRef(({
|
|
48
48
|
id,
|
|
49
49
|
ariaLabel = "Menu",
|
|
@@ -19,15 +19,15 @@ const StyledPopup = /* @__PURE__ */ _styled(Popup, process.env.NODE_ENV === "pro
|
|
|
19
19
|
label: "StyledPopup"
|
|
20
20
|
})("background-color:", ({
|
|
21
21
|
theme
|
|
22
|
-
}) => theme.colors.
|
|
22
|
+
}) => theme.colors.other.elevation.background.raised, ";box-shadow:", ({
|
|
23
23
|
theme
|
|
24
|
-
}) => theme.shadows.
|
|
24
|
+
}) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`, ";padding:0;&[data-has-arrow='true']{&::after{border-color:", ({
|
|
25
25
|
theme
|
|
26
|
-
}) => theme.colors.
|
|
26
|
+
}) => theme.colors.other.elevation.background.raised, " transparent transparent transparent;}}width:", ({
|
|
27
27
|
size
|
|
28
28
|
}) => SIZES[size], ";max-width:none;padding:", ({
|
|
29
29
|
theme
|
|
30
|
-
}) => `${theme.space["0.25"]} 0`, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01lbnVWMi9pbmRleC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBeUNnQyIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9NZW51VjIvaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7XG4gIEJ1dHRvbkhUTUxBdHRyaWJ1dGVzLFxuICBDb21wb25lbnRQcm9wcyxcbiAgTW91c2VFdmVudCxcbiAgUmVhY3RFbGVtZW50LFxuICBSZWFjdE5vZGUsXG4gIFJlZixcbn0gZnJvbSAncmVhY3QnXG5pbXBvcnQge1xuICBjbG9uZUVsZW1lbnQsXG4gIGZvcndhcmRSZWYsXG4gIGlzVmFsaWRFbGVtZW50LFxuICB1c2VJZCxcbiAgdXNlSW1wZXJhdGl2ZUhhbmRsZSxcbiAgdXNlUmVmLFxuICB1c2VTdGF0ZSxcbn0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBQb3B1cCB9IGZyb20gJy4uL1BvcHVwJ1xuaW1wb3J0IHsgU3RhY2sgfSBmcm9tICcuLi9TdGFjaydcbmltcG9ydCB7IEdyb3VwIH0gZnJvbSAnLi9Hcm91cCdcbmltcG9ydCBJdGVtIGZyb20gJy4vSXRlbSdcblxuY29uc3QgU0laRVMgPSB7XG4gIHNtYWxsOiAnMTgwcHgnLFxuICBtZWRpdW06ICcyODBweCcsXG4gIGxhcmdlOiAnMzgwcHgnLFxufVxuXG5leHBvcnQgdHlwZSBEaXNjbG9zdXJlUHJvcHMgPSB7IHZpc2libGU6IGJvb2xlYW4gfVxuXG50eXBlIERpc2Nsb3N1cmVFbGVtZW50ID1cbiAgfCAoKFxuICAgICAgZGlzY2xvc3VyZTogRGlzY2xvc3VyZVByb3BzLFxuICAgICkgPT4gUmVhY3RFbGVtZW50PEJ1dHRvbkhUTUxBdHRyaWJ1dGVzPEhUTUxCdXR0b25FbGVtZW50Pj4pXG4gIHwgKFJlYWN0RWxlbWVudDxCdXR0b25IVE1MQXR0cmlidXRlczxIVE1MQnV0dG9uRWxlbWVudD4+ICYge1xuICAgICAgcmVmPzogUmVmPEhUTUxCdXR0b25FbGVtZW50PlxuICAgIH0pXG5cbmNvbnN0IFN0eWxlZFBvcHVwID0gc3R5bGVkKFBvcHVwLCB7XG4gIHNob3VsZEZvcndhcmRQcm9wOiBwcm9wID0+ICFbJ3NpemUnXS5pbmNsdWRlcyhwcm9wKSxcbn0pPHsgc2l6ZToga2V5b2YgdHlwZW9mIFNJWkVTIH0+YFxuICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+
|
|
30
|
+
}) => `${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/index.tsx"],"names":[],"mappings":"AAyCgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  ComponentProps,\n  MouseEvent,\n  ReactElement,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useId,\n  useImperativeHandle,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Group } from './Group'\nimport Item from './Item'\n\nconst SIZES = {\n  small: '180px',\n  medium: '280px',\n  large: '380px',\n}\n\nexport type DisclosureProps = { visible: boolean }\n\ntype DisclosureElement =\n  | ((\n      disclosure: DisclosureProps,\n    ) => ReactElement<ButtonHTMLAttributes<HTMLButtonElement>>)\n  | (ReactElement<ButtonHTMLAttributes<HTMLButtonElement>> & {\n      ref?: Ref<HTMLButtonElement>\n    })\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: keyof typeof SIZES }>`\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  width: ${({ size }) => SIZES[size]};\n  max-width: none;\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\ntype ChildMenuProps = {\n  toggle: () => void\n}\n\ntype MenuProps = {\n  id?: string\n  ariaLabel?: string\n  children?: ReactNode | (({ toggle }: ChildMenuProps) => ReactNode)\n  className?: string\n  disclosure: DisclosureElement\n  hasArrow?: boolean\n  visible?: boolean\n  'data-testid'?: string\n  maxHeight?: string\n  /**\n   * @deprecated: use `size` instead\n   */\n  maxWidth?: string\n  /**\n   * By default, the portal target is children container or document.body if children is a function. You can override this\n   * behavior by setting a portalTarget prop.\n   */\n  portalTarget?: HTMLElement\n  size?: keyof typeof SIZES\n  /**\n   * The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.\n   * If set to `hover`, the menu will open when the user hovers over the disclosure.\n   */\n  triggerMethod?: 'click' | 'hover'\n} & Pick<ComponentProps<typeof Popup>, 'placement' | 'dynamicDomRendering'>\n\nconst FwdMenu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      visible = false,\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget,\n      size = 'small',\n      triggerMethod = 'click',\n      dynamicDomRendering,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const [isVisible, setIsVisible] = useState(visible)\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    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={() => setIsVisible(false)}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '480px'}\n        maxWidth={maxWidth}\n        size={size}\n        text={\n          <MenuList data-testid={dataTestId} className={className} role=\"menu\">\n            {typeof children === 'function'\n              ? children({ toggle: () => setIsVisible(!isVisible) })\n              : children}\n          </MenuList>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n\n/**\n * A menu is a widget that offers a list of choices to the user, such as a set of actions or functions.\n * A menu is usually opened, or made visible, by activating a menu button, choosing an item in a menu that opens a\n * sub menu, or by invoking a command, such as `Shift + F10` on Windows, that opens a context specific menu.\n * When a user activates a choice in a menu, the menu usually closes unless the choice opened a submenu.\n */\nexport const MenuV2 = Object.assign(FwdMenu, { Item, Group })\n"]} */"));
|
|
31
31
|
const MenuList = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
|
|
32
32
|
target: "e1jn11gg0"
|
|
33
33
|
} : {
|
|
@@ -35,11 +35,11 @@ const MenuList = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "produc
|
|
|
35
35
|
label: "MenuList"
|
|
36
36
|
})("overflow-y:auto;overflow-x:hidden;&:after,&:before{border:solid transparent;border-width:9px;content:' ';height:0;width:0;position:absolute;pointer-events:none;}&:after{border-color:transparent;}&:before{border-color:transparent;}background-color:", ({
|
|
37
37
|
theme
|
|
38
|
-
}) => theme.colors.
|
|
38
|
+
}) => theme.colors.other.elevation.background.raised, ";color:", ({
|
|
39
39
|
theme
|
|
40
40
|
}) => theme.colors.neutral.text, ";border-radius:", ({
|
|
41
41
|
theme
|
|
42
|
-
}) => theme.radii.default, ";position:relative;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01lbnVWMi9pbmRleC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBMEQ4QiIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9NZW51VjIvaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7XG4gIEJ1dHRvbkhUTUxBdHRyaWJ1dGVzLFxuICBDb21wb25lbnRQcm9wcyxcbiAgTW91c2VFdmVudCxcbiAgUmVhY3RFbGVtZW50LFxuICBSZWFjdE5vZGUsXG4gIFJlZixcbn0gZnJvbSAncmVhY3QnXG5pbXBvcnQge1xuICBjbG9uZUVsZW1lbnQsXG4gIGZvcndhcmRSZWYsXG4gIGlzVmFsaWRFbGVtZW50LFxuICB1c2VJZCxcbiAgdXNlSW1wZXJhdGl2ZUhhbmRsZSxcbiAgdXNlUmVmLFxuICB1c2VTdGF0ZSxcbn0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBQb3B1cCB9IGZyb20gJy4uL1BvcHVwJ1xuaW1wb3J0IHsgU3RhY2sgfSBmcm9tICcuLi9TdGFjaydcbmltcG9ydCB7IEdyb3VwIH0gZnJvbSAnLi9Hcm91cCdcbmltcG9ydCBJdGVtIGZyb20gJy4vSXRlbSdcblxuY29uc3QgU0laRVMgPSB7XG4gIHNtYWxsOiAnMTgwcHgnLFxuICBtZWRpdW06ICcyODBweCcsXG4gIGxhcmdlOiAnMzgwcHgnLFxufVxuXG5leHBvcnQgdHlwZSBEaXNjbG9zdXJlUHJvcHMgPSB7IHZpc2libGU6IGJvb2xlYW4gfVxuXG50eXBlIERpc2Nsb3N1cmVFbGVtZW50ID1cbiAgfCAoKFxuICAgICAgZGlzY2xvc3VyZTogRGlzY2xvc3VyZVByb3BzLFxuICAgICkgPT4gUmVhY3RFbGVtZW50PEJ1dHRvbkhUTUxBdHRyaWJ1dGVzPEhUTUxCdXR0b25FbGVtZW50Pj4pXG4gIHwgKFJlYWN0RWxlbWVudDxCdXR0b25IVE1MQXR0cmlidXRlczxIVE1MQnV0dG9uRWxlbWVudD4+ICYge1xuICAgICAgcmVmPzogUmVmPEhUTUxCdXR0b25FbGVtZW50PlxuICAgIH0pXG5cbmNvbnN0IFN0eWxlZFBvcHVwID0gc3R5bGVkKFBvcHVwLCB7XG4gIHNob3VsZEZvcndhcmRQcm9wOiBwcm9wID0+ICFbJ3NpemUnXS5pbmNsdWRlcyhwcm9wKSxcbn0pPHsgc2l6ZToga2V5b2YgdHlwZW9mIFNJWkVTIH0+YFxuICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+
|
|
42
|
+
}) => 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/index.tsx"],"names":[],"mappings":"AA0D8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  ComponentProps,\n  MouseEvent,\n  ReactElement,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useId,\n  useImperativeHandle,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Group } from './Group'\nimport Item from './Item'\n\nconst SIZES = {\n  small: '180px',\n  medium: '280px',\n  large: '380px',\n}\n\nexport type DisclosureProps = { visible: boolean }\n\ntype DisclosureElement =\n  | ((\n      disclosure: DisclosureProps,\n    ) => ReactElement<ButtonHTMLAttributes<HTMLButtonElement>>)\n  | (ReactElement<ButtonHTMLAttributes<HTMLButtonElement>> & {\n      ref?: Ref<HTMLButtonElement>\n    })\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: keyof typeof SIZES }>`\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  width: ${({ size }) => SIZES[size]};\n  max-width: none;\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\ntype ChildMenuProps = {\n  toggle: () => void\n}\n\ntype MenuProps = {\n  id?: string\n  ariaLabel?: string\n  children?: ReactNode | (({ toggle }: ChildMenuProps) => ReactNode)\n  className?: string\n  disclosure: DisclosureElement\n  hasArrow?: boolean\n  visible?: boolean\n  'data-testid'?: string\n  maxHeight?: string\n  /**\n   * @deprecated: use `size` instead\n   */\n  maxWidth?: string\n  /**\n   * By default, the portal target is children container or document.body if children is a function. You can override this\n   * behavior by setting a portalTarget prop.\n   */\n  portalTarget?: HTMLElement\n  size?: keyof typeof SIZES\n  /**\n   * The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.\n   * If set to `hover`, the menu will open when the user hovers over the disclosure.\n   */\n  triggerMethod?: 'click' | 'hover'\n} & Pick<ComponentProps<typeof Popup>, 'placement' | 'dynamicDomRendering'>\n\nconst FwdMenu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      visible = false,\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget,\n      size = 'small',\n      triggerMethod = 'click',\n      dynamicDomRendering,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const [isVisible, setIsVisible] = useState(visible)\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    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={() => setIsVisible(false)}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '480px'}\n        maxWidth={maxWidth}\n        size={size}\n        text={\n          <MenuList data-testid={dataTestId} className={className} role=\"menu\">\n            {typeof children === 'function'\n              ? children({ toggle: () => setIsVisible(!isVisible) })\n              : children}\n          </MenuList>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n\n/**\n * A menu is a widget that offers a list of choices to the user, such as a set of actions or functions.\n * A menu is usually opened, or made visible, by activating a menu button, choosing an item in a menu that opens a\n * sub menu, or by invoking a command, such as `Shift + F10` on Windows, that opens a context specific menu.\n * When a user activates a choice in a menu, the menu usually closes unless the choice opened a submenu.\n */\nexport const MenuV2 = Object.assign(FwdMenu, { Item, Group })\n"]} */"));
|
|
43
43
|
const FwdMenu = forwardRef(({
|
|
44
44
|
id,
|
|
45
45
|
ariaLabel = "Menu",
|
|
@@ -16,7 +16,7 @@ const StyledBackdrop = /* @__PURE__ */ _styled__default.default("div", process.e
|
|
|
16
16
|
theme
|
|
17
17
|
}) => theme.colors.overlay, ";z-index:1;&[data-open='true']{padding:", ({
|
|
18
18
|
theme
|
|
19
|
-
}) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
19
|
+
}) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}" + (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/Dialog.tsx"],"names":[],"mappings":"AAY2D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/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 } from 'react'\nimport { createPortal } from 'react-dom'\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\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\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n}\n\nexport const StyledDialog = styled.dialog<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}px;\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}px;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n`\n\nexport const Dialog = ({\n  children,\n  open,\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 containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    if (open) {\n      document.body.appendChild(element)\n    }\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [open])\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    if (open) {\n      dialogRef.current?.focus()\n    }\n  }, [open])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (open && preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll, open])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // Stop click to prevent unexpected dialog close\n  const stopClick: MouseEventHandler = 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      if (hideOnClickOutside) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\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  // Prevent default behaviour on Escape\n  const stopCancel: ReactEventHandler = event => {\n    event.preventDefault()\n    event.stopPropagation()\n  }\n\n  return open\n    ? createPortal(\n        <StyledBackdrop\n          data-open={open}\n          onClick={handleClose}\n          className={backdropClassName}\n          css={backdropCss}\n          data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n          onFocus={stopFocus}\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={open}\n            onClick={stopClick}\n            onCancel={stopCancel}\n            onClose={stopCancel}\n            aria-modal\n            ref={dialogRef}\n            tabIndex={0}\n          >\n            {open ? children : null}\n          </StyledDialog>\n        </StyledBackdrop>,\n        containerRef.current,\n      )\n    : null\n}\n"]} */"));
|
|
20
20
|
const StyledDialog = /* @__PURE__ */ _styled__default.default("dialog", process.env.NODE_ENV === "production" ? {
|
|
21
21
|
target: "e1cqen9h0"
|
|
22
22
|
} : {
|
|
@@ -24,13 +24,13 @@ const StyledDialog = /* @__PURE__ */ _styled__default.default("dialog", process.
|
|
|
24
24
|
label: "StyledDialog"
|
|
25
25
|
})("background-color:", ({
|
|
26
26
|
theme
|
|
27
|
-
}) => theme.colors.
|
|
27
|
+
}) => theme.colors.other.elevation.background.overlay, ";position:relative;border-radius:", ({
|
|
28
28
|
theme
|
|
29
29
|
}) => theme.radii.default, ";border:0;padding:", ({
|
|
30
30
|
theme
|
|
31
31
|
}) => theme.space["3"], ";width:", constants.MODAL_WIDTH.medium, "px;box-shadow:", ({
|
|
32
32
|
theme
|
|
33
|
-
}) => theme.shadows.
|
|
33
|
+
}) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`, ";", Object.entries(constants.MODAL_WIDTH).map(([size, value]) => `
|
|
34
34
|
&[data-size="${size}"] {
|
|
35
35
|
width: ${value}px;
|
|
36
36
|
}
|
|
@@ -38,7 +38,7 @@ const StyledDialog = /* @__PURE__ */ _styled__default.default("dialog", process.
|
|
|
38
38
|
&[data-placement="${placement}"] {
|
|
39
39
|
${value}
|
|
40
40
|
}
|
|
41
|
-
`), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01vZGFsL0RpYWxvZy50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBc0M0RCIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9Nb2RhbC9EaWFsb2cudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7XG4gIEZvY3VzRXZlbnRIYW5kbGVyLFxuICBLZXlib2FyZEV2ZW50SGFuZGxlcixcbiAgTW91c2VFdmVudEhhbmRsZXIsXG4gIFJlYWN0RXZlbnRIYW5kbGVyLFxufSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZUNhbGxiYWNrLCB1c2VFZmZlY3QsIHVzZVJlZiB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgY3JlYXRlUG9ydGFsIH0gZnJvbSAncmVhY3QtZG9tJ1xuaW1wb3J0IHsgTU9EQUxfUExBQ0VNRU5ULCBNT0RBTF9XSURUSCB9IGZyb20gJy4vY29uc3RhbnRzJ1xuaW1wb3J0IHR5cGUgeyBEaWFsb2dQcm9wcywgTW9kYWxQbGFjZW1lbnQsIE1vZGFsU2l6ZSB9IGZyb20gJy4vdHlwZXMnXG5cbmNvbnN0IFN0eWxlZEJhY2tkcm9wID0gc3R5bGVkLmRpdjx7ICdkYXRhLW9wZW4nOiBib29sZWFuIH0+
|
|
41
|
+
`), ";" + (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/Dialog.tsx"],"names":[],"mappings":"AAsC4D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/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 } from 'react'\nimport { createPortal } from 'react-dom'\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\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\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n}\n\nexport const StyledDialog = styled.dialog<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}px;\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}px;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n`\n\nexport const Dialog = ({\n  children,\n  open,\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 containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    if (open) {\n      document.body.appendChild(element)\n    }\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [open])\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    if (open) {\n      dialogRef.current?.focus()\n    }\n  }, [open])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (open && preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll, open])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // Stop click to prevent unexpected dialog close\n  const stopClick: MouseEventHandler = 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      if (hideOnClickOutside) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\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  // Prevent default behaviour on Escape\n  const stopCancel: ReactEventHandler = event => {\n    event.preventDefault()\n    event.stopPropagation()\n  }\n\n  return open\n    ? createPortal(\n        <StyledBackdrop\n          data-open={open}\n          onClick={handleClose}\n          className={backdropClassName}\n          css={backdropCss}\n          data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n          onFocus={stopFocus}\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={open}\n            onClick={stopClick}\n            onCancel={stopCancel}\n            onClose={stopCancel}\n            aria-modal\n            ref={dialogRef}\n            tabIndex={0}\n          >\n            {open ? children : null}\n          </StyledDialog>\n        </StyledBackdrop>,\n        containerRef.current,\n      )\n    : null\n}\n"]} */"));
|
|
42
42
|
const Dialog = ({
|
|
43
43
|
children,
|
|
44
44
|
open,
|
|
@@ -12,7 +12,7 @@ const StyledBackdrop = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "
|
|
|
12
12
|
theme
|
|
13
13
|
}) => theme.colors.overlay, ";z-index:1;&[data-open='true']{padding:", ({
|
|
14
14
|
theme
|
|
15
|
-
}) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
15
|
+
}) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}" + (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/Dialog.tsx"],"names":[],"mappings":"AAY2D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/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 } from 'react'\nimport { createPortal } from 'react-dom'\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\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\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n}\n\nexport const StyledDialog = styled.dialog<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}px;\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}px;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n`\n\nexport const Dialog = ({\n  children,\n  open,\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 containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    if (open) {\n      document.body.appendChild(element)\n    }\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [open])\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    if (open) {\n      dialogRef.current?.focus()\n    }\n  }, [open])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (open && preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll, open])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // Stop click to prevent unexpected dialog close\n  const stopClick: MouseEventHandler = 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      if (hideOnClickOutside) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\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  // Prevent default behaviour on Escape\n  const stopCancel: ReactEventHandler = event => {\n    event.preventDefault()\n    event.stopPropagation()\n  }\n\n  return open\n    ? createPortal(\n        <StyledBackdrop\n          data-open={open}\n          onClick={handleClose}\n          className={backdropClassName}\n          css={backdropCss}\n          data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n          onFocus={stopFocus}\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={open}\n            onClick={stopClick}\n            onCancel={stopCancel}\n            onClose={stopCancel}\n            aria-modal\n            ref={dialogRef}\n            tabIndex={0}\n          >\n            {open ? children : null}\n          </StyledDialog>\n        </StyledBackdrop>,\n        containerRef.current,\n      )\n    : null\n}\n"]} */"));
|
|
16
16
|
const StyledDialog = /* @__PURE__ */ _styled("dialog", process.env.NODE_ENV === "production" ? {
|
|
17
17
|
target: "e1cqen9h0"
|
|
18
18
|
} : {
|
|
@@ -20,13 +20,13 @@ const StyledDialog = /* @__PURE__ */ _styled("dialog", process.env.NODE_ENV ===
|
|
|
20
20
|
label: "StyledDialog"
|
|
21
21
|
})("background-color:", ({
|
|
22
22
|
theme
|
|
23
|
-
}) => theme.colors.
|
|
23
|
+
}) => theme.colors.other.elevation.background.overlay, ";position:relative;border-radius:", ({
|
|
24
24
|
theme
|
|
25
25
|
}) => theme.radii.default, ";border:0;padding:", ({
|
|
26
26
|
theme
|
|
27
27
|
}) => theme.space["3"], ";width:", MODAL_WIDTH.medium, "px;box-shadow:", ({
|
|
28
28
|
theme
|
|
29
|
-
}) => theme.shadows.
|
|
29
|
+
}) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`, ";", Object.entries(MODAL_WIDTH).map(([size, value]) => `
|
|
30
30
|
&[data-size="${size}"] {
|
|
31
31
|
width: ${value}px;
|
|
32
32
|
}
|
|
@@ -34,7 +34,7 @@ const StyledDialog = /* @__PURE__ */ _styled("dialog", process.env.NODE_ENV ===
|
|
|
34
34
|
&[data-placement="${placement}"] {
|
|
35
35
|
${value}
|
|
36
36
|
}
|
|
37
|
-
`), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01vZGFsL0RpYWxvZy50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBc0M0RCIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9Nb2RhbC9EaWFsb2cudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7XG4gIEZvY3VzRXZlbnRIYW5kbGVyLFxuICBLZXlib2FyZEV2ZW50SGFuZGxlcixcbiAgTW91c2VFdmVudEhhbmRsZXIsXG4gIFJlYWN0RXZlbnRIYW5kbGVyLFxufSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZUNhbGxiYWNrLCB1c2VFZmZlY3QsIHVzZVJlZiB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgY3JlYXRlUG9ydGFsIH0gZnJvbSAncmVhY3QtZG9tJ1xuaW1wb3J0IHsgTU9EQUxfUExBQ0VNRU5ULCBNT0RBTF9XSURUSCB9IGZyb20gJy4vY29uc3RhbnRzJ1xuaW1wb3J0IHR5cGUgeyBEaWFsb2dQcm9wcywgTW9kYWxQbGFjZW1lbnQsIE1vZGFsU2l6ZSB9IGZyb20gJy4vdHlwZXMnXG5cbmNvbnN0IFN0eWxlZEJhY2tkcm9wID0gc3R5bGVkLmRpdjx7ICdkYXRhLW9wZW4nOiBib29sZWFuIH0+
|
|
37
|
+
`), ";" + (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/Dialog.tsx"],"names":[],"mappings":"AAsC4D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/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 } from 'react'\nimport { createPortal } from 'react-dom'\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\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\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n}\n\nexport const StyledDialog = styled.dialog<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}px;\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}px;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n`\n\nexport const Dialog = ({\n  children,\n  open,\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 containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    if (open) {\n      document.body.appendChild(element)\n    }\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [open])\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    if (open) {\n      dialogRef.current?.focus()\n    }\n  }, [open])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (open && preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll, open])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // Stop click to prevent unexpected dialog close\n  const stopClick: MouseEventHandler = 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      if (hideOnClickOutside) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\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  // Prevent default behaviour on Escape\n  const stopCancel: ReactEventHandler = event => {\n    event.preventDefault()\n    event.stopPropagation()\n  }\n\n  return open\n    ? createPortal(\n        <StyledBackdrop\n          data-open={open}\n          onClick={handleClose}\n          className={backdropClassName}\n          css={backdropCss}\n          data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n          onFocus={stopFocus}\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={open}\n            onClick={stopClick}\n            onCancel={stopCancel}\n            onClose={stopCancel}\n            aria-modal\n            ref={dialogRef}\n            tabIndex={0}\n          >\n            {open ? children : null}\n          </StyledDialog>\n        </StyledBackdrop>,\n        containerRef.current,\n      )\n    : null\n}\n"]} */"));
|
|
38
38
|
const Dialog = ({
|
|
39
39
|
children,
|
|
40
40
|
open,
|