@ultraviolet/ui 1.60.0 → 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.
|
@@ -13,11 +13,11 @@ const StyledExpandable = /* @__PURE__ */ _styled__default.default("div", process
|
|
|
13
13
|
shouldForwardProp: (prop) => !["animationDuration"].includes(prop),
|
|
14
14
|
target: "e5hc7t70",
|
|
15
15
|
label: "StyledExpandable"
|
|
16
|
-
})("transition:max-height
|
|
16
|
+
})('&[data-is-animated="true"]{transition:max-height ', ({
|
|
17
17
|
animationDuration
|
|
18
18
|
}) => animationDuration, "ms ease-out,opacity ", ({
|
|
19
19
|
animationDuration
|
|
20
|
-
}) => animationDuration, "ms ease-out;height:auto;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
20
|
+
}) => 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 */"));
|
|
21
21
|
const Expandable = ({
|
|
22
22
|
children,
|
|
23
23
|
opened,
|
|
@@ -29,6 +29,7 @@ const Expandable = ({
|
|
|
29
29
|
const [height, setHeight] = React.useState(null);
|
|
30
30
|
const transitionTimer = React.useRef();
|
|
31
31
|
const ref = React.useRef(null);
|
|
32
|
+
const shouldBeAnimated = animationDuration > 0;
|
|
32
33
|
React.useEffect(() => {
|
|
33
34
|
if (ref.current) {
|
|
34
35
|
setHeight(ref.current.scrollHeight ?? 0);
|
|
@@ -38,35 +39,49 @@ const Expandable = ({
|
|
|
38
39
|
if (opened && ref.current && height) {
|
|
39
40
|
ref.current.style.maxHeight = `${height}px`;
|
|
40
41
|
ref.current.style.visibility = "";
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
ref.current
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
42
|
+
if (shouldBeAnimated) {
|
|
43
|
+
transitionTimer.current = setTimeout(() => {
|
|
44
|
+
if (ref.current) {
|
|
45
|
+
ref.current.style.maxHeight = "initial";
|
|
46
|
+
ref.current.style.overflow = "visible";
|
|
47
|
+
ref.current.style.visibility = "";
|
|
48
|
+
}
|
|
49
|
+
}, ANIMATION_DURATION);
|
|
50
|
+
} else {
|
|
51
|
+
ref.current.style.maxHeight = "initial";
|
|
52
|
+
ref.current.style.overflow = "visible";
|
|
53
|
+
ref.current.style.visibility = "";
|
|
54
|
+
}
|
|
48
55
|
} else {
|
|
49
56
|
clearTimeout(transitionTimer.current);
|
|
50
57
|
if (ref.current && height) {
|
|
51
58
|
ref.current.style.maxHeight = `${height}px`;
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
ref.current
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
ref.current
|
|
59
|
-
|
|
60
|
-
|
|
59
|
+
if (shouldBeAnimated) {
|
|
60
|
+
transitionTimer.current = setTimeout(() => {
|
|
61
|
+
if (ref.current) {
|
|
62
|
+
ref.current.style.maxHeight = `${minHeight}px`;
|
|
63
|
+
ref.current.style.overflow = "hidden";
|
|
64
|
+
setTimeout(() => {
|
|
65
|
+
if (ref.current && !minHeight) {
|
|
66
|
+
ref.current.style.visibility = "hidden";
|
|
67
|
+
}
|
|
68
|
+
}, ANIMATION_DURATION);
|
|
69
|
+
}
|
|
70
|
+
}, 0);
|
|
71
|
+
} else {
|
|
72
|
+
ref.current.style.maxHeight = `${minHeight}px`;
|
|
73
|
+
ref.current.style.overflow = "hidden";
|
|
74
|
+
if (!minHeight) {
|
|
75
|
+
ref.current.style.visibility = "hidden";
|
|
61
76
|
}
|
|
62
|
-
}
|
|
77
|
+
}
|
|
63
78
|
}
|
|
64
79
|
}
|
|
65
80
|
return () => {
|
|
66
81
|
clearTimeout(transitionTimer.current);
|
|
67
82
|
};
|
|
68
|
-
}, [height, minHeight, opened]);
|
|
69
|
-
return /* @__PURE__ */ jsxRuntime.jsx(StyledExpandable, { "data-testid": dataTestId, ref, className, animationDuration, children });
|
|
83
|
+
}, [animationDuration, height, minHeight, opened, shouldBeAnimated]);
|
|
84
|
+
return /* @__PURE__ */ jsxRuntime.jsx(StyledExpandable, { "data-testid": dataTestId, ref, className, animationDuration, "data-is-animated": shouldBeAnimated, children });
|
|
70
85
|
};
|
|
71
86
|
exports.Expandable = Expandable;
|
|
72
87
|
exports.StyledExpandable = StyledExpandable;
|
|
@@ -14,6 +14,9 @@ type ExpandableProps = {
|
|
|
14
14
|
minHeight?: number;
|
|
15
15
|
className?: string;
|
|
16
16
|
'data-testid'?: string;
|
|
17
|
+
/**
|
|
18
|
+
* The duration of the animation in ms. If set to 0, the animation will be disabled.
|
|
19
|
+
*/
|
|
17
20
|
animationDuration?: number;
|
|
18
21
|
};
|
|
19
22
|
export declare const StyledExpandable: import("@emotion/styled").StyledComponent<{
|
|
@@ -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,
|
|
@@ -21,13 +21,13 @@ const StyledPopup = /* @__PURE__ */ _styled__default.default(index.Popup, proces
|
|
|
21
21
|
theme
|
|
22
22
|
}) => theme.colors.other.elevation.background.raised, ";box-shadow:", ({
|
|
23
23
|
theme
|
|
24
|
-
}) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[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/SearchInput/index.tsx"],"names":[],"mappings":"AAwBiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\nimport { isClientSide } from '../../helpers/isClientSide'\nimport { Popup } from '../Popup'\nimport {\n  BasicPrefixStack,\n  BasicSuffixStack,\n  StyledInput,\n  TextInputV2,\n} from '../TextInputV2'\nimport { KeyGroup } from './KeyGroup'\nimport type { SearchInputProps } from './types'\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  text-align: initial;\n  min-width: 610px;\n  padding: ${({ theme }) => `${theme.space['2']} ${theme.space['1']}`};\n  background: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n`\n\nconst StyledTextInputV2 = styled(TextInputV2)`\n  ${BasicPrefixStack} {\n    border: none;\n  }\n\n  ${StyledInput} {\n    padding: 0;\n  }\n\n  ${BasicSuffixStack} {\n    border: none;\n  }\n`\n\n/**\n * SearchInput is a component that allows users to search for items. It is a combination of a TextInputV2 and a Popup. The Popup is used to display search results.\n * Children of the SearchInput component can be a function that receives an object with the following properties:\n * - `searchTerms`: the current search terms\n * - `isOpen`: a boolean indicating if the popup is open\n * - `toggleIsOpen`: a function to toggle the popup\n */\nexport const SearchInput = forwardRef(\n  (\n    {\n      placeholder,\n      label,\n      labelDescription,\n      loading,\n      size,\n      popupPlacement,\n      threshold = 0,\n      children,\n      onSearch,\n      onClose,\n      'data-testid': dataTestId,\n      shortcut = false,\n      error,\n      disabled,\n      className,\n      minLength,\n      maxLength,\n      tooltip,\n      onFocus,\n      onBlur,\n      name,\n      id,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      'aria-labelledby': ariaLabelledby,\n      readOnly,\n      required,\n      autoFocus,\n      autoComplete,\n      onKeyDown,\n      role,\n    }: SearchInputProps,\n    ref: Ref<HTMLInputElement>,\n  ) => {\n    const focusedLinkIndex = useRef(0)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const [containerWidth, setContainerWidth] = useState(0)\n    const [searchTerms, setSearchTerms] = useState('')\n    const [isMacOS, setIsMacOS] = useState(false)\n    const [keyPressed, setKeyPressed] = useState<string[]>([])\n    const [isOpen, toggleIsOpen] = useReducer(state => !state, false)\n    const innerSearchInputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(\n      ref,\n      () => innerSearchInputRef.current as HTMLInputElement,\n    )\n\n    const content =\n      typeof children === 'function'\n        ? children({ searchTerms, isOpen, toggleIsOpen })\n        : children\n\n    const resizeSearchBar = () => {\n      if (popupRef.current) {\n        setContainerWidth(popupRef.current.getBoundingClientRect().width)\n      }\n    }\n\n    const handleNavigation = (event: KeyboardEvent) => {\n      const links = [...(popupRef.current?.querySelectorAll('a') ?? [])]\n\n      if (\n        links.length > 0 &&\n        focusedLinkIndex.current >= 0 &&\n        focusedLinkIndex.current <= links.length\n      ) {\n        if (event.key === 'ArrowUp') {\n          if (focusedLinkIndex.current - 1 < 0) {\n            focusedLinkIndex.current = links.length - 1\n          } else {\n            focusedLinkIndex.current -= 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n\n        if (event.key === 'ArrowDown') {\n          if (focusedLinkIndex.current + 1 >= links.length) {\n            focusedLinkIndex.current = 0\n          } else {\n            focusedLinkIndex.current += 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n      }\n    }\n\n    useEffect(() => {\n      document.addEventListener('keyup', handleNavigation)\n\n      return () => document.removeEventListener('keyup', handleNavigation)\n    }, [])\n\n    useEffect(() => {\n      resizeSearchBar()\n\n      window.addEventListener('resize', resizeSearchBar)\n\n      return () => window.removeEventListener('resize', resizeSearchBar)\n    }, [])\n\n    const onSearchCallback = (localValue: string) => {\n      setSearchTerms(localValue)\n\n      try {\n        onSearch(localValue)\n        if (localValue.length >= threshold && !isOpen) {\n          toggleIsOpen()\n        }\n      } catch {\n        toggleIsOpen()\n      }\n    }\n\n    const onCloseCallback = () => {\n      onClose?.()\n      if (isOpen) {\n        toggleIsOpen()\n      }\n    }\n\n    useEffect(() => {\n      if (isClientSide) {\n        // We need to check if window is defined to avoid SSR issues\n        setIsMacOS(navigator.userAgent.includes('Mac'))\n      }\n    }, [])\n\n    const handleKeyPressed = useCallback(\n      (event: KeyboardEvent) => {\n        const { ctrlKey, metaKey, key } = event\n        setKeyPressed([...keyPressed, key.toUpperCase()])\n\n        if (typeof shortcut === 'boolean') {\n          if (\n            (key === 'k' || key === 'K') &&\n            ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        } else {\n          const uppercaseShortcut = shortcut.map(s => s.toUpperCase())\n\n          if (\n            JSON.stringify([...keyPressed, key.toUpperCase()]) ===\n            JSON.stringify(uppercaseShortcut)\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        }\n      },\n      [keyPressed, shortcut, isMacOS],\n    )\n\n    const handleKeyReleased = useCallback(\n      (event: KeyboardEvent) => {\n        const { key } = event\n        setKeyPressed(keyPressed.filter(k => k !== key.toUpperCase()))\n      },\n      [keyPressed],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleKeyPressed)\n        document.body.addEventListener('keyup', handleKeyReleased)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleKeyPressed)\n        document.body.removeEventListener('keyup', handleKeyReleased)\n      }\n    }, [shortcut, disabled, handleKeyPressed, handleKeyReleased])\n\n    const keys = useMemo(() => {\n      if (typeof shortcut === 'boolean') {\n        return [isMacOS ? '⌘' : 'Ctrl', 'K']\n      }\n\n      const filteredKey = shortcut.map(key => {\n        if (key === 'Meta') {\n          return '⌘'\n        }\n\n        if (key === 'Control') {\n          return 'Ctrl'\n        }\n\n        return key\n      })\n\n      return filteredKey\n    }, [isMacOS, shortcut])\n\n    return (\n      <div style={{ width: '100%' }}>\n        <StyledPopup\n          data-testid={`popup-${dataTestId}`}\n          role=\"dialog\"\n          visible={isOpen}\n          onClose={onCloseCallback}\n          placement={popupPlacement}\n          maxWidth={containerWidth}\n          hideOnClickOutside\n          hasArrow={false}\n          innerRef={popupRef}\n          text={content}\n          maxHeight={410}\n          debounceDelay={0}\n        >\n          <StyledTextInputV2\n            ref={innerSearchInputRef}\n            prefix={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup disabled={disabled} keys={keys} />\n              ) : undefined\n            }\n            data-testid={dataTestId}\n            error={error}\n            value={searchTerms}\n            size={size}\n            label={label}\n            placeholder={placeholder}\n            loading={loading}\n            onChange={onSearchCallback}\n            clearable\n            disabled={disabled}\n            className={className}\n            aria-atomic={ariaAtomic}\n            autoComplete={autoComplete}\n            aria-labelledby={ariaLabelledby}\n            aria-live={ariaLive}\n            id={id}\n            name={name}\n            readOnly={readOnly}\n            required={required}\n            autoFocus={autoFocus}\n            maxLength={maxLength}\n            minLength={minLength}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            onKeyDown={onKeyDown}\n            role={role}\n            tooltip={tooltip}\n            labelDescription={labelDescription}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
|
|
24
|
+
}) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[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/SearchInput/index.tsx"],"names":[],"mappings":"AAwBiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\nimport { isClientSide } from '../../helpers/isClientSide'\nimport { Popup } from '../Popup'\nimport {\n  BasicPrefixStack,\n  BasicSuffixStack,\n  StyledInput,\n  TextInputV2,\n} from '../TextInputV2'\nimport { KeyGroup } from './KeyGroup'\nimport type { SearchInputProps } from './types'\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  text-align: initial;\n  min-width: 610px;\n  padding: ${({ theme }) => `${theme.space['2']} ${theme.space['1']}`};\n  background: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n`\n\nconst StyledTextInputV2 = styled(TextInputV2)`\n  ${BasicPrefixStack} {\n    border: none;\n  }\n\n  ${StyledInput} {\n    padding: 0;\n  }\n\n  ${BasicSuffixStack} {\n    border: none;\n  }\n`\n\n/**\n * SearchInput is a component that allows users to search for items. It is a combination of a TextInputV2 and a Popup. The Popup is used to display search results.\n * Children of the SearchInput component can be a function that receives an object with the following properties:\n * - `searchTerms`: the current search terms\n * - `isOpen`: a boolean indicating if the popup is open\n * - `toggleIsOpen`: a function to toggle the popup\n */\nexport const SearchInput = forwardRef(\n  (\n    {\n      placeholder,\n      label,\n      labelDescription,\n      loading,\n      size,\n      popupPlacement,\n      threshold = 0,\n      children,\n      onSearch,\n      onClose,\n      'data-testid': dataTestId,\n      shortcut = false,\n      error,\n      disabled,\n      className,\n      minLength,\n      maxLength,\n      tooltip,\n      onFocus,\n      onBlur,\n      name,\n      id,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      'aria-labelledby': ariaLabelledby,\n      readOnly,\n      required,\n      autoFocus,\n      autoComplete,\n      onKeyDown,\n      role,\n    }: SearchInputProps,\n    ref: Ref<HTMLInputElement>,\n  ) => {\n    const focusedLinkIndex = useRef(0)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const [containerWidth, setContainerWidth] = useState(0)\n    const [searchTerms, setSearchTerms] = useState('')\n    const [isMacOS, setIsMacOS] = useState(false)\n    const [keyPressed, setKeyPressed] = useState<string[]>([])\n    const [isOpen, toggleIsOpen] = useReducer(state => !state, false)\n    const innerSearchInputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(\n      ref,\n      () => innerSearchInputRef.current as HTMLInputElement,\n    )\n\n    const content =\n      typeof children === 'function'\n        ? children({ searchTerms, isOpen, toggleIsOpen })\n        : children\n\n    const resizeSearchBar = () => {\n      if (popupRef.current) {\n        setContainerWidth(popupRef.current.getBoundingClientRect().width)\n      }\n    }\n\n    const handleNavigation = (event: KeyboardEvent) => {\n      const links = [...(popupRef.current?.querySelectorAll('a') ?? [])]\n\n      if (\n        links.length > 0 &&\n        focusedLinkIndex.current >= 0 &&\n        focusedLinkIndex.current <= links.length\n      ) {\n        if (event.key === 'ArrowUp') {\n          if (focusedLinkIndex.current - 1 < 0) {\n            focusedLinkIndex.current = links.length - 1\n          } else {\n            focusedLinkIndex.current -= 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n\n        if (event.key === 'ArrowDown') {\n          if (focusedLinkIndex.current + 1 >= links.length) {\n            focusedLinkIndex.current = 0\n          } else {\n            focusedLinkIndex.current += 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n      }\n    }\n\n    useEffect(() => {\n      document.addEventListener('keyup', handleNavigation)\n\n      return () => document.removeEventListener('keyup', handleNavigation)\n    }, [])\n\n    useEffect(() => {\n      resizeSearchBar()\n\n      window.addEventListener('resize', resizeSearchBar)\n\n      return () => window.removeEventListener('resize', resizeSearchBar)\n    }, [])\n\n    const onSearchCallback = (localValue: string) => {\n      setSearchTerms(localValue)\n\n      try {\n        onSearch(localValue)\n        if (localValue.length >= threshold && !isOpen) {\n          toggleIsOpen()\n        }\n      } catch {\n        toggleIsOpen()\n      }\n    }\n\n    const onCloseCallback = () => {\n      onClose?.()\n      if (isOpen) {\n        toggleIsOpen()\n      }\n    }\n\n    useEffect(() => {\n      if (isClientSide) {\n        // We need to check if window is defined to avoid SSR issues\n        setIsMacOS(navigator.userAgent.includes('Mac'))\n      }\n    }, [])\n\n    const handleKeyPressed = useCallback(\n      (event: KeyboardEvent) => {\n        if (!(event instanceof KeyboardEvent)) {\n          return\n        }\n\n        const { ctrlKey, metaKey, key } = event\n        setKeyPressed([...keyPressed, key.toUpperCase()])\n\n        if (typeof shortcut === 'boolean') {\n          if (\n            (key === 'k' || key === 'K') &&\n            ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        } else {\n          const uppercaseShortcut = shortcut.map(s => s.toUpperCase())\n\n          if (\n            JSON.stringify([...keyPressed, key.toUpperCase()]) ===\n            JSON.stringify(uppercaseShortcut)\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        }\n      },\n      [keyPressed, shortcut, isMacOS],\n    )\n\n    const handleKeyReleased = useCallback(\n      (event: KeyboardEvent) => {\n        const { key } = event\n        setKeyPressed(keyPressed.filter(k => k !== key.toUpperCase()))\n      },\n      [keyPressed],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleKeyPressed)\n        document.body.addEventListener('keyup', handleKeyReleased)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleKeyPressed)\n        document.body.removeEventListener('keyup', handleKeyReleased)\n      }\n    }, [shortcut, disabled, handleKeyPressed, handleKeyReleased])\n\n    const keys = useMemo(() => {\n      if (typeof shortcut === 'boolean') {\n        return [isMacOS ? '⌘' : 'Ctrl', 'K']\n      }\n\n      const filteredKey = shortcut.map(key => {\n        if (key === 'Meta') {\n          return '⌘'\n        }\n\n        if (key === 'Control') {\n          return 'Ctrl'\n        }\n\n        return key\n      })\n\n      return filteredKey\n    }, [isMacOS, shortcut])\n\n    return (\n      <div style={{ width: '100%' }}>\n        <StyledPopup\n          data-testid={`popup-${dataTestId}`}\n          role=\"dialog\"\n          visible={isOpen}\n          onClose={onCloseCallback}\n          placement={popupPlacement}\n          maxWidth={containerWidth}\n          hideOnClickOutside\n          hasArrow={false}\n          innerRef={popupRef}\n          text={content}\n          maxHeight={410}\n          debounceDelay={0}\n        >\n          <StyledTextInputV2\n            ref={innerSearchInputRef}\n            prefix={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup disabled={disabled} keys={keys} />\n              ) : undefined\n            }\n            data-testid={dataTestId}\n            error={error}\n            value={searchTerms}\n            size={size}\n            label={label}\n            placeholder={placeholder}\n            loading={loading}\n            onChange={onSearchCallback}\n            clearable\n            disabled={disabled}\n            className={className}\n            aria-atomic={ariaAtomic}\n            autoComplete={autoComplete}\n            aria-labelledby={ariaLabelledby}\n            aria-live={ariaLive}\n            id={id}\n            name={name}\n            readOnly={readOnly}\n            required={required}\n            autoFocus={autoFocus}\n            maxLength={maxLength}\n            minLength={minLength}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            onKeyDown={onKeyDown}\n            role={role}\n            tooltip={tooltip}\n            labelDescription={labelDescription}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
|
|
25
25
|
const StyledTextInputV2 = /* @__PURE__ */ _styled__default.default(index$1.TextInputV2, process.env.NODE_ENV === "production" ? {
|
|
26
26
|
target: "eefux4u0"
|
|
27
27
|
} : {
|
|
28
28
|
target: "eefux4u0",
|
|
29
29
|
label: "StyledTextInputV2"
|
|
30
|
-
})(index$1.BasicPrefixStack, "{border:none;}", index$1.StyledInput, "{padding:0;}", index$1.BasicSuffixStack, "{border:none;}" + (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/SearchInput/index.tsx"],"names":[],"mappings":"AAiC6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\nimport { isClientSide } from '../../helpers/isClientSide'\nimport { Popup } from '../Popup'\nimport {\n  BasicPrefixStack,\n  BasicSuffixStack,\n  StyledInput,\n  TextInputV2,\n} from '../TextInputV2'\nimport { KeyGroup } from './KeyGroup'\nimport type { SearchInputProps } from './types'\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  text-align: initial;\n  min-width: 610px;\n  padding: ${({ theme }) => `${theme.space['2']} ${theme.space['1']}`};\n  background: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n`\n\nconst StyledTextInputV2 = styled(TextInputV2)`\n  ${BasicPrefixStack} {\n    border: none;\n  }\n\n  ${StyledInput} {\n    padding: 0;\n  }\n\n  ${BasicSuffixStack} {\n    border: none;\n  }\n`\n\n/**\n * SearchInput is a component that allows users to search for items. It is a combination of a TextInputV2 and a Popup. The Popup is used to display search results.\n * Children of the SearchInput component can be a function that receives an object with the following properties:\n * - `searchTerms`: the current search terms\n * - `isOpen`: a boolean indicating if the popup is open\n * - `toggleIsOpen`: a function to toggle the popup\n */\nexport const SearchInput = forwardRef(\n  (\n    {\n      placeholder,\n      label,\n      labelDescription,\n      loading,\n      size,\n      popupPlacement,\n      threshold = 0,\n      children,\n      onSearch,\n      onClose,\n      'data-testid': dataTestId,\n      shortcut = false,\n      error,\n      disabled,\n      className,\n      minLength,\n      maxLength,\n      tooltip,\n      onFocus,\n      onBlur,\n      name,\n      id,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      'aria-labelledby': ariaLabelledby,\n      readOnly,\n      required,\n      autoFocus,\n      autoComplete,\n      onKeyDown,\n      role,\n    }: SearchInputProps,\n    ref: Ref<HTMLInputElement>,\n  ) => {\n    const focusedLinkIndex = useRef(0)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const [containerWidth, setContainerWidth] = useState(0)\n    const [searchTerms, setSearchTerms] = useState('')\n    const [isMacOS, setIsMacOS] = useState(false)\n    const [keyPressed, setKeyPressed] = useState<string[]>([])\n    const [isOpen, toggleIsOpen] = useReducer(state => !state, false)\n    const innerSearchInputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(\n      ref,\n      () => innerSearchInputRef.current as HTMLInputElement,\n    )\n\n    const content =\n      typeof children === 'function'\n        ? children({ searchTerms, isOpen, toggleIsOpen })\n        : children\n\n    const resizeSearchBar = () => {\n      if (popupRef.current) {\n        setContainerWidth(popupRef.current.getBoundingClientRect().width)\n      }\n    }\n\n    const handleNavigation = (event: KeyboardEvent) => {\n      const links = [...(popupRef.current?.querySelectorAll('a') ?? [])]\n\n      if (\n        links.length > 0 &&\n        focusedLinkIndex.current >= 0 &&\n        focusedLinkIndex.current <= links.length\n      ) {\n        if (event.key === 'ArrowUp') {\n          if (focusedLinkIndex.current - 1 < 0) {\n            focusedLinkIndex.current = links.length - 1\n          } else {\n            focusedLinkIndex.current -= 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n\n        if (event.key === 'ArrowDown') {\n          if (focusedLinkIndex.current + 1 >= links.length) {\n            focusedLinkIndex.current = 0\n          } else {\n            focusedLinkIndex.current += 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n      }\n    }\n\n    useEffect(() => {\n      document.addEventListener('keyup', handleNavigation)\n\n      return () => document.removeEventListener('keyup', handleNavigation)\n    }, [])\n\n    useEffect(() => {\n      resizeSearchBar()\n\n      window.addEventListener('resize', resizeSearchBar)\n\n      return () => window.removeEventListener('resize', resizeSearchBar)\n    }, [])\n\n    const onSearchCallback = (localValue: string) => {\n      setSearchTerms(localValue)\n\n      try {\n        onSearch(localValue)\n        if (localValue.length >= threshold && !isOpen) {\n          toggleIsOpen()\n        }\n      } catch {\n        toggleIsOpen()\n      }\n    }\n\n    const onCloseCallback = () => {\n      onClose?.()\n      if (isOpen) {\n        toggleIsOpen()\n      }\n    }\n\n    useEffect(() => {\n      if (isClientSide) {\n        // We need to check if window is defined to avoid SSR issues\n        setIsMacOS(navigator.userAgent.includes('Mac'))\n      }\n    }, [])\n\n    const handleKeyPressed = useCallback(\n      (event: KeyboardEvent) => {\n        const { ctrlKey, metaKey, key } = event\n        setKeyPressed([...keyPressed, key.toUpperCase()])\n\n        if (typeof shortcut === 'boolean') {\n          if (\n            (key === 'k' || key === 'K') &&\n            ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        } else {\n          const uppercaseShortcut = shortcut.map(s => s.toUpperCase())\n\n          if (\n            JSON.stringify([...keyPressed, key.toUpperCase()]) ===\n            JSON.stringify(uppercaseShortcut)\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        }\n      },\n      [keyPressed, shortcut, isMacOS],\n    )\n\n    const handleKeyReleased = useCallback(\n      (event: KeyboardEvent) => {\n        const { key } = event\n        setKeyPressed(keyPressed.filter(k => k !== key.toUpperCase()))\n      },\n      [keyPressed],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleKeyPressed)\n        document.body.addEventListener('keyup', handleKeyReleased)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleKeyPressed)\n        document.body.removeEventListener('keyup', handleKeyReleased)\n      }\n    }, [shortcut, disabled, handleKeyPressed, handleKeyReleased])\n\n    const keys = useMemo(() => {\n      if (typeof shortcut === 'boolean') {\n        return [isMacOS ? '⌘' : 'Ctrl', 'K']\n      }\n\n      const filteredKey = shortcut.map(key => {\n        if (key === 'Meta') {\n          return '⌘'\n        }\n\n        if (key === 'Control') {\n          return 'Ctrl'\n        }\n\n        return key\n      })\n\n      return filteredKey\n    }, [isMacOS, shortcut])\n\n    return (\n      <div style={{ width: '100%' }}>\n        <StyledPopup\n          data-testid={`popup-${dataTestId}`}\n          role=\"dialog\"\n          visible={isOpen}\n          onClose={onCloseCallback}\n          placement={popupPlacement}\n          maxWidth={containerWidth}\n          hideOnClickOutside\n          hasArrow={false}\n          innerRef={popupRef}\n          text={content}\n          maxHeight={410}\n          debounceDelay={0}\n        >\n          <StyledTextInputV2\n            ref={innerSearchInputRef}\n            prefix={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup disabled={disabled} keys={keys} />\n              ) : undefined\n            }\n            data-testid={dataTestId}\n            error={error}\n            value={searchTerms}\n            size={size}\n            label={label}\n            placeholder={placeholder}\n            loading={loading}\n            onChange={onSearchCallback}\n            clearable\n            disabled={disabled}\n            className={className}\n            aria-atomic={ariaAtomic}\n            autoComplete={autoComplete}\n            aria-labelledby={ariaLabelledby}\n            aria-live={ariaLive}\n            id={id}\n            name={name}\n            readOnly={readOnly}\n            required={required}\n            autoFocus={autoFocus}\n            maxLength={maxLength}\n            minLength={minLength}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            onKeyDown={onKeyDown}\n            role={role}\n            tooltip={tooltip}\n            labelDescription={labelDescription}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
|
|
30
|
+
})(index$1.BasicPrefixStack, "{border:none;}", index$1.StyledInput, "{padding:0;}", index$1.BasicSuffixStack, "{border:none;}" + (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/SearchInput/index.tsx"],"names":[],"mappings":"AAiC6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\nimport { isClientSide } from '../../helpers/isClientSide'\nimport { Popup } from '../Popup'\nimport {\n  BasicPrefixStack,\n  BasicSuffixStack,\n  StyledInput,\n  TextInputV2,\n} from '../TextInputV2'\nimport { KeyGroup } from './KeyGroup'\nimport type { SearchInputProps } from './types'\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  text-align: initial;\n  min-width: 610px;\n  padding: ${({ theme }) => `${theme.space['2']} ${theme.space['1']}`};\n  background: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n`\n\nconst StyledTextInputV2 = styled(TextInputV2)`\n  ${BasicPrefixStack} {\n    border: none;\n  }\n\n  ${StyledInput} {\n    padding: 0;\n  }\n\n  ${BasicSuffixStack} {\n    border: none;\n  }\n`\n\n/**\n * SearchInput is a component that allows users to search for items. It is a combination of a TextInputV2 and a Popup. The Popup is used to display search results.\n * Children of the SearchInput component can be a function that receives an object with the following properties:\n * - `searchTerms`: the current search terms\n * - `isOpen`: a boolean indicating if the popup is open\n * - `toggleIsOpen`: a function to toggle the popup\n */\nexport const SearchInput = forwardRef(\n  (\n    {\n      placeholder,\n      label,\n      labelDescription,\n      loading,\n      size,\n      popupPlacement,\n      threshold = 0,\n      children,\n      onSearch,\n      onClose,\n      'data-testid': dataTestId,\n      shortcut = false,\n      error,\n      disabled,\n      className,\n      minLength,\n      maxLength,\n      tooltip,\n      onFocus,\n      onBlur,\n      name,\n      id,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      'aria-labelledby': ariaLabelledby,\n      readOnly,\n      required,\n      autoFocus,\n      autoComplete,\n      onKeyDown,\n      role,\n    }: SearchInputProps,\n    ref: Ref<HTMLInputElement>,\n  ) => {\n    const focusedLinkIndex = useRef(0)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const [containerWidth, setContainerWidth] = useState(0)\n    const [searchTerms, setSearchTerms] = useState('')\n    const [isMacOS, setIsMacOS] = useState(false)\n    const [keyPressed, setKeyPressed] = useState<string[]>([])\n    const [isOpen, toggleIsOpen] = useReducer(state => !state, false)\n    const innerSearchInputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(\n      ref,\n      () => innerSearchInputRef.current as HTMLInputElement,\n    )\n\n    const content =\n      typeof children === 'function'\n        ? children({ searchTerms, isOpen, toggleIsOpen })\n        : children\n\n    const resizeSearchBar = () => {\n      if (popupRef.current) {\n        setContainerWidth(popupRef.current.getBoundingClientRect().width)\n      }\n    }\n\n    const handleNavigation = (event: KeyboardEvent) => {\n      const links = [...(popupRef.current?.querySelectorAll('a') ?? [])]\n\n      if (\n        links.length > 0 &&\n        focusedLinkIndex.current >= 0 &&\n        focusedLinkIndex.current <= links.length\n      ) {\n        if (event.key === 'ArrowUp') {\n          if (focusedLinkIndex.current - 1 < 0) {\n            focusedLinkIndex.current = links.length - 1\n          } else {\n            focusedLinkIndex.current -= 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n\n        if (event.key === 'ArrowDown') {\n          if (focusedLinkIndex.current + 1 >= links.length) {\n            focusedLinkIndex.current = 0\n          } else {\n            focusedLinkIndex.current += 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n      }\n    }\n\n    useEffect(() => {\n      document.addEventListener('keyup', handleNavigation)\n\n      return () => document.removeEventListener('keyup', handleNavigation)\n    }, [])\n\n    useEffect(() => {\n      resizeSearchBar()\n\n      window.addEventListener('resize', resizeSearchBar)\n\n      return () => window.removeEventListener('resize', resizeSearchBar)\n    }, [])\n\n    const onSearchCallback = (localValue: string) => {\n      setSearchTerms(localValue)\n\n      try {\n        onSearch(localValue)\n        if (localValue.length >= threshold && !isOpen) {\n          toggleIsOpen()\n        }\n      } catch {\n        toggleIsOpen()\n      }\n    }\n\n    const onCloseCallback = () => {\n      onClose?.()\n      if (isOpen) {\n        toggleIsOpen()\n      }\n    }\n\n    useEffect(() => {\n      if (isClientSide) {\n        // We need to check if window is defined to avoid SSR issues\n        setIsMacOS(navigator.userAgent.includes('Mac'))\n      }\n    }, [])\n\n    const handleKeyPressed = useCallback(\n      (event: KeyboardEvent) => {\n        if (!(event instanceof KeyboardEvent)) {\n          return\n        }\n\n        const { ctrlKey, metaKey, key } = event\n        setKeyPressed([...keyPressed, key.toUpperCase()])\n\n        if (typeof shortcut === 'boolean') {\n          if (\n            (key === 'k' || key === 'K') &&\n            ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        } else {\n          const uppercaseShortcut = shortcut.map(s => s.toUpperCase())\n\n          if (\n            JSON.stringify([...keyPressed, key.toUpperCase()]) ===\n            JSON.stringify(uppercaseShortcut)\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        }\n      },\n      [keyPressed, shortcut, isMacOS],\n    )\n\n    const handleKeyReleased = useCallback(\n      (event: KeyboardEvent) => {\n        const { key } = event\n        setKeyPressed(keyPressed.filter(k => k !== key.toUpperCase()))\n      },\n      [keyPressed],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleKeyPressed)\n        document.body.addEventListener('keyup', handleKeyReleased)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleKeyPressed)\n        document.body.removeEventListener('keyup', handleKeyReleased)\n      }\n    }, [shortcut, disabled, handleKeyPressed, handleKeyReleased])\n\n    const keys = useMemo(() => {\n      if (typeof shortcut === 'boolean') {\n        return [isMacOS ? '⌘' : 'Ctrl', 'K']\n      }\n\n      const filteredKey = shortcut.map(key => {\n        if (key === 'Meta') {\n          return '⌘'\n        }\n\n        if (key === 'Control') {\n          return 'Ctrl'\n        }\n\n        return key\n      })\n\n      return filteredKey\n    }, [isMacOS, shortcut])\n\n    return (\n      <div style={{ width: '100%' }}>\n        <StyledPopup\n          data-testid={`popup-${dataTestId}`}\n          role=\"dialog\"\n          visible={isOpen}\n          onClose={onCloseCallback}\n          placement={popupPlacement}\n          maxWidth={containerWidth}\n          hideOnClickOutside\n          hasArrow={false}\n          innerRef={popupRef}\n          text={content}\n          maxHeight={410}\n          debounceDelay={0}\n        >\n          <StyledTextInputV2\n            ref={innerSearchInputRef}\n            prefix={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup disabled={disabled} keys={keys} />\n              ) : undefined\n            }\n            data-testid={dataTestId}\n            error={error}\n            value={searchTerms}\n            size={size}\n            label={label}\n            placeholder={placeholder}\n            loading={loading}\n            onChange={onSearchCallback}\n            clearable\n            disabled={disabled}\n            className={className}\n            aria-atomic={ariaAtomic}\n            autoComplete={autoComplete}\n            aria-labelledby={ariaLabelledby}\n            aria-live={ariaLive}\n            id={id}\n            name={name}\n            readOnly={readOnly}\n            required={required}\n            autoFocus={autoFocus}\n            maxLength={maxLength}\n            minLength={minLength}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            onKeyDown={onKeyDown}\n            role={role}\n            tooltip={tooltip}\n            labelDescription={labelDescription}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
|
|
31
31
|
const SearchInput = React.forwardRef(({
|
|
32
32
|
placeholder,
|
|
33
33
|
label,
|
|
@@ -133,6 +133,9 @@ const SearchInput = React.forwardRef(({
|
|
|
133
133
|
}
|
|
134
134
|
}, []);
|
|
135
135
|
const handleKeyPressed = React.useCallback((event) => {
|
|
136
|
+
if (!(event instanceof KeyboardEvent)) {
|
|
137
|
+
return;
|
|
138
|
+
}
|
|
136
139
|
const {
|
|
137
140
|
ctrlKey,
|
|
138
141
|
metaKey,
|
|
@@ -17,13 +17,13 @@ const StyledPopup = /* @__PURE__ */ _styled(Popup, process.env.NODE_ENV === "pro
|
|
|
17
17
|
theme
|
|
18
18
|
}) => theme.colors.other.elevation.background.raised, ";box-shadow:", ({
|
|
19
19
|
theme
|
|
20
|
-
}) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[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/SearchInput/index.tsx"],"names":[],"mappings":"AAwBiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\nimport { isClientSide } from '../../helpers/isClientSide'\nimport { Popup } from '../Popup'\nimport {\n  BasicPrefixStack,\n  BasicSuffixStack,\n  StyledInput,\n  TextInputV2,\n} from '../TextInputV2'\nimport { KeyGroup } from './KeyGroup'\nimport type { SearchInputProps } from './types'\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  text-align: initial;\n  min-width: 610px;\n  padding: ${({ theme }) => `${theme.space['2']} ${theme.space['1']}`};\n  background: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n`\n\nconst StyledTextInputV2 = styled(TextInputV2)`\n  ${BasicPrefixStack} {\n    border: none;\n  }\n\n  ${StyledInput} {\n    padding: 0;\n  }\n\n  ${BasicSuffixStack} {\n    border: none;\n  }\n`\n\n/**\n * SearchInput is a component that allows users to search for items. It is a combination of a TextInputV2 and a Popup. The Popup is used to display search results.\n * Children of the SearchInput component can be a function that receives an object with the following properties:\n * - `searchTerms`: the current search terms\n * - `isOpen`: a boolean indicating if the popup is open\n * - `toggleIsOpen`: a function to toggle the popup\n */\nexport const SearchInput = forwardRef(\n  (\n    {\n      placeholder,\n      label,\n      labelDescription,\n      loading,\n      size,\n      popupPlacement,\n      threshold = 0,\n      children,\n      onSearch,\n      onClose,\n      'data-testid': dataTestId,\n      shortcut = false,\n      error,\n      disabled,\n      className,\n      minLength,\n      maxLength,\n      tooltip,\n      onFocus,\n      onBlur,\n      name,\n      id,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      'aria-labelledby': ariaLabelledby,\n      readOnly,\n      required,\n      autoFocus,\n      autoComplete,\n      onKeyDown,\n      role,\n    }: SearchInputProps,\n    ref: Ref<HTMLInputElement>,\n  ) => {\n    const focusedLinkIndex = useRef(0)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const [containerWidth, setContainerWidth] = useState(0)\n    const [searchTerms, setSearchTerms] = useState('')\n    const [isMacOS, setIsMacOS] = useState(false)\n    const [keyPressed, setKeyPressed] = useState<string[]>([])\n    const [isOpen, toggleIsOpen] = useReducer(state => !state, false)\n    const innerSearchInputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(\n      ref,\n      () => innerSearchInputRef.current as HTMLInputElement,\n    )\n\n    const content =\n      typeof children === 'function'\n        ? children({ searchTerms, isOpen, toggleIsOpen })\n        : children\n\n    const resizeSearchBar = () => {\n      if (popupRef.current) {\n        setContainerWidth(popupRef.current.getBoundingClientRect().width)\n      }\n    }\n\n    const handleNavigation = (event: KeyboardEvent) => {\n      const links = [...(popupRef.current?.querySelectorAll('a') ?? [])]\n\n      if (\n        links.length > 0 &&\n        focusedLinkIndex.current >= 0 &&\n        focusedLinkIndex.current <= links.length\n      ) {\n        if (event.key === 'ArrowUp') {\n          if (focusedLinkIndex.current - 1 < 0) {\n            focusedLinkIndex.current = links.length - 1\n          } else {\n            focusedLinkIndex.current -= 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n\n        if (event.key === 'ArrowDown') {\n          if (focusedLinkIndex.current + 1 >= links.length) {\n            focusedLinkIndex.current = 0\n          } else {\n            focusedLinkIndex.current += 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n      }\n    }\n\n    useEffect(() => {\n      document.addEventListener('keyup', handleNavigation)\n\n      return () => document.removeEventListener('keyup', handleNavigation)\n    }, [])\n\n    useEffect(() => {\n      resizeSearchBar()\n\n      window.addEventListener('resize', resizeSearchBar)\n\n      return () => window.removeEventListener('resize', resizeSearchBar)\n    }, [])\n\n    const onSearchCallback = (localValue: string) => {\n      setSearchTerms(localValue)\n\n      try {\n        onSearch(localValue)\n        if (localValue.length >= threshold && !isOpen) {\n          toggleIsOpen()\n        }\n      } catch {\n        toggleIsOpen()\n      }\n    }\n\n    const onCloseCallback = () => {\n      onClose?.()\n      if (isOpen) {\n        toggleIsOpen()\n      }\n    }\n\n    useEffect(() => {\n      if (isClientSide) {\n        // We need to check if window is defined to avoid SSR issues\n        setIsMacOS(navigator.userAgent.includes('Mac'))\n      }\n    }, [])\n\n    const handleKeyPressed = useCallback(\n      (event: KeyboardEvent) => {\n        const { ctrlKey, metaKey, key } = event\n        setKeyPressed([...keyPressed, key.toUpperCase()])\n\n        if (typeof shortcut === 'boolean') {\n          if (\n            (key === 'k' || key === 'K') &&\n            ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        } else {\n          const uppercaseShortcut = shortcut.map(s => s.toUpperCase())\n\n          if (\n            JSON.stringify([...keyPressed, key.toUpperCase()]) ===\n            JSON.stringify(uppercaseShortcut)\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        }\n      },\n      [keyPressed, shortcut, isMacOS],\n    )\n\n    const handleKeyReleased = useCallback(\n      (event: KeyboardEvent) => {\n        const { key } = event\n        setKeyPressed(keyPressed.filter(k => k !== key.toUpperCase()))\n      },\n      [keyPressed],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleKeyPressed)\n        document.body.addEventListener('keyup', handleKeyReleased)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleKeyPressed)\n        document.body.removeEventListener('keyup', handleKeyReleased)\n      }\n    }, [shortcut, disabled, handleKeyPressed, handleKeyReleased])\n\n    const keys = useMemo(() => {\n      if (typeof shortcut === 'boolean') {\n        return [isMacOS ? '⌘' : 'Ctrl', 'K']\n      }\n\n      const filteredKey = shortcut.map(key => {\n        if (key === 'Meta') {\n          return '⌘'\n        }\n\n        if (key === 'Control') {\n          return 'Ctrl'\n        }\n\n        return key\n      })\n\n      return filteredKey\n    }, [isMacOS, shortcut])\n\n    return (\n      <div style={{ width: '100%' }}>\n        <StyledPopup\n          data-testid={`popup-${dataTestId}`}\n          role=\"dialog\"\n          visible={isOpen}\n          onClose={onCloseCallback}\n          placement={popupPlacement}\n          maxWidth={containerWidth}\n          hideOnClickOutside\n          hasArrow={false}\n          innerRef={popupRef}\n          text={content}\n          maxHeight={410}\n          debounceDelay={0}\n        >\n          <StyledTextInputV2\n            ref={innerSearchInputRef}\n            prefix={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup disabled={disabled} keys={keys} />\n              ) : undefined\n            }\n            data-testid={dataTestId}\n            error={error}\n            value={searchTerms}\n            size={size}\n            label={label}\n            placeholder={placeholder}\n            loading={loading}\n            onChange={onSearchCallback}\n            clearable\n            disabled={disabled}\n            className={className}\n            aria-atomic={ariaAtomic}\n            autoComplete={autoComplete}\n            aria-labelledby={ariaLabelledby}\n            aria-live={ariaLive}\n            id={id}\n            name={name}\n            readOnly={readOnly}\n            required={required}\n            autoFocus={autoFocus}\n            maxLength={maxLength}\n            minLength={minLength}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            onKeyDown={onKeyDown}\n            role={role}\n            tooltip={tooltip}\n            labelDescription={labelDescription}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
|
|
20
|
+
}) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[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/SearchInput/index.tsx"],"names":[],"mappings":"AAwBiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\nimport { isClientSide } from '../../helpers/isClientSide'\nimport { Popup } from '../Popup'\nimport {\n  BasicPrefixStack,\n  BasicSuffixStack,\n  StyledInput,\n  TextInputV2,\n} from '../TextInputV2'\nimport { KeyGroup } from './KeyGroup'\nimport type { SearchInputProps } from './types'\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  text-align: initial;\n  min-width: 610px;\n  padding: ${({ theme }) => `${theme.space['2']} ${theme.space['1']}`};\n  background: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n`\n\nconst StyledTextInputV2 = styled(TextInputV2)`\n  ${BasicPrefixStack} {\n    border: none;\n  }\n\n  ${StyledInput} {\n    padding: 0;\n  }\n\n  ${BasicSuffixStack} {\n    border: none;\n  }\n`\n\n/**\n * SearchInput is a component that allows users to search for items. It is a combination of a TextInputV2 and a Popup. The Popup is used to display search results.\n * Children of the SearchInput component can be a function that receives an object with the following properties:\n * - `searchTerms`: the current search terms\n * - `isOpen`: a boolean indicating if the popup is open\n * - `toggleIsOpen`: a function to toggle the popup\n */\nexport const SearchInput = forwardRef(\n  (\n    {\n      placeholder,\n      label,\n      labelDescription,\n      loading,\n      size,\n      popupPlacement,\n      threshold = 0,\n      children,\n      onSearch,\n      onClose,\n      'data-testid': dataTestId,\n      shortcut = false,\n      error,\n      disabled,\n      className,\n      minLength,\n      maxLength,\n      tooltip,\n      onFocus,\n      onBlur,\n      name,\n      id,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      'aria-labelledby': ariaLabelledby,\n      readOnly,\n      required,\n      autoFocus,\n      autoComplete,\n      onKeyDown,\n      role,\n    }: SearchInputProps,\n    ref: Ref<HTMLInputElement>,\n  ) => {\n    const focusedLinkIndex = useRef(0)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const [containerWidth, setContainerWidth] = useState(0)\n    const [searchTerms, setSearchTerms] = useState('')\n    const [isMacOS, setIsMacOS] = useState(false)\n    const [keyPressed, setKeyPressed] = useState<string[]>([])\n    const [isOpen, toggleIsOpen] = useReducer(state => !state, false)\n    const innerSearchInputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(\n      ref,\n      () => innerSearchInputRef.current as HTMLInputElement,\n    )\n\n    const content =\n      typeof children === 'function'\n        ? children({ searchTerms, isOpen, toggleIsOpen })\n        : children\n\n    const resizeSearchBar = () => {\n      if (popupRef.current) {\n        setContainerWidth(popupRef.current.getBoundingClientRect().width)\n      }\n    }\n\n    const handleNavigation = (event: KeyboardEvent) => {\n      const links = [...(popupRef.current?.querySelectorAll('a') ?? [])]\n\n      if (\n        links.length > 0 &&\n        focusedLinkIndex.current >= 0 &&\n        focusedLinkIndex.current <= links.length\n      ) {\n        if (event.key === 'ArrowUp') {\n          if (focusedLinkIndex.current - 1 < 0) {\n            focusedLinkIndex.current = links.length - 1\n          } else {\n            focusedLinkIndex.current -= 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n\n        if (event.key === 'ArrowDown') {\n          if (focusedLinkIndex.current + 1 >= links.length) {\n            focusedLinkIndex.current = 0\n          } else {\n            focusedLinkIndex.current += 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n      }\n    }\n\n    useEffect(() => {\n      document.addEventListener('keyup', handleNavigation)\n\n      return () => document.removeEventListener('keyup', handleNavigation)\n    }, [])\n\n    useEffect(() => {\n      resizeSearchBar()\n\n      window.addEventListener('resize', resizeSearchBar)\n\n      return () => window.removeEventListener('resize', resizeSearchBar)\n    }, [])\n\n    const onSearchCallback = (localValue: string) => {\n      setSearchTerms(localValue)\n\n      try {\n        onSearch(localValue)\n        if (localValue.length >= threshold && !isOpen) {\n          toggleIsOpen()\n        }\n      } catch {\n        toggleIsOpen()\n      }\n    }\n\n    const onCloseCallback = () => {\n      onClose?.()\n      if (isOpen) {\n        toggleIsOpen()\n      }\n    }\n\n    useEffect(() => {\n      if (isClientSide) {\n        // We need to check if window is defined to avoid SSR issues\n        setIsMacOS(navigator.userAgent.includes('Mac'))\n      }\n    }, [])\n\n    const handleKeyPressed = useCallback(\n      (event: KeyboardEvent) => {\n        if (!(event instanceof KeyboardEvent)) {\n          return\n        }\n\n        const { ctrlKey, metaKey, key } = event\n        setKeyPressed([...keyPressed, key.toUpperCase()])\n\n        if (typeof shortcut === 'boolean') {\n          if (\n            (key === 'k' || key === 'K') &&\n            ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        } else {\n          const uppercaseShortcut = shortcut.map(s => s.toUpperCase())\n\n          if (\n            JSON.stringify([...keyPressed, key.toUpperCase()]) ===\n            JSON.stringify(uppercaseShortcut)\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        }\n      },\n      [keyPressed, shortcut, isMacOS],\n    )\n\n    const handleKeyReleased = useCallback(\n      (event: KeyboardEvent) => {\n        const { key } = event\n        setKeyPressed(keyPressed.filter(k => k !== key.toUpperCase()))\n      },\n      [keyPressed],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleKeyPressed)\n        document.body.addEventListener('keyup', handleKeyReleased)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleKeyPressed)\n        document.body.removeEventListener('keyup', handleKeyReleased)\n      }\n    }, [shortcut, disabled, handleKeyPressed, handleKeyReleased])\n\n    const keys = useMemo(() => {\n      if (typeof shortcut === 'boolean') {\n        return [isMacOS ? '⌘' : 'Ctrl', 'K']\n      }\n\n      const filteredKey = shortcut.map(key => {\n        if (key === 'Meta') {\n          return '⌘'\n        }\n\n        if (key === 'Control') {\n          return 'Ctrl'\n        }\n\n        return key\n      })\n\n      return filteredKey\n    }, [isMacOS, shortcut])\n\n    return (\n      <div style={{ width: '100%' }}>\n        <StyledPopup\n          data-testid={`popup-${dataTestId}`}\n          role=\"dialog\"\n          visible={isOpen}\n          onClose={onCloseCallback}\n          placement={popupPlacement}\n          maxWidth={containerWidth}\n          hideOnClickOutside\n          hasArrow={false}\n          innerRef={popupRef}\n          text={content}\n          maxHeight={410}\n          debounceDelay={0}\n        >\n          <StyledTextInputV2\n            ref={innerSearchInputRef}\n            prefix={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup disabled={disabled} keys={keys} />\n              ) : undefined\n            }\n            data-testid={dataTestId}\n            error={error}\n            value={searchTerms}\n            size={size}\n            label={label}\n            placeholder={placeholder}\n            loading={loading}\n            onChange={onSearchCallback}\n            clearable\n            disabled={disabled}\n            className={className}\n            aria-atomic={ariaAtomic}\n            autoComplete={autoComplete}\n            aria-labelledby={ariaLabelledby}\n            aria-live={ariaLive}\n            id={id}\n            name={name}\n            readOnly={readOnly}\n            required={required}\n            autoFocus={autoFocus}\n            maxLength={maxLength}\n            minLength={minLength}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            onKeyDown={onKeyDown}\n            role={role}\n            tooltip={tooltip}\n            labelDescription={labelDescription}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
|
|
21
21
|
const StyledTextInputV2 = /* @__PURE__ */ _styled(TextInputV2, process.env.NODE_ENV === "production" ? {
|
|
22
22
|
target: "eefux4u0"
|
|
23
23
|
} : {
|
|
24
24
|
target: "eefux4u0",
|
|
25
25
|
label: "StyledTextInputV2"
|
|
26
|
-
})(BasicPrefixStack, "{border:none;}", StyledInput, "{padding:0;}", BasicSuffixStack, "{border:none;}" + (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/SearchInput/index.tsx"],"names":[],"mappings":"AAiC6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\nimport { isClientSide } from '../../helpers/isClientSide'\nimport { Popup } from '../Popup'\nimport {\n  BasicPrefixStack,\n  BasicSuffixStack,\n  StyledInput,\n  TextInputV2,\n} from '../TextInputV2'\nimport { KeyGroup } from './KeyGroup'\nimport type { SearchInputProps } from './types'\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  text-align: initial;\n  min-width: 610px;\n  padding: ${({ theme }) => `${theme.space['2']} ${theme.space['1']}`};\n  background: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n`\n\nconst StyledTextInputV2 = styled(TextInputV2)`\n  ${BasicPrefixStack} {\n    border: none;\n  }\n\n  ${StyledInput} {\n    padding: 0;\n  }\n\n  ${BasicSuffixStack} {\n    border: none;\n  }\n`\n\n/**\n * SearchInput is a component that allows users to search for items. It is a combination of a TextInputV2 and a Popup. The Popup is used to display search results.\n * Children of the SearchInput component can be a function that receives an object with the following properties:\n * - `searchTerms`: the current search terms\n * - `isOpen`: a boolean indicating if the popup is open\n * - `toggleIsOpen`: a function to toggle the popup\n */\nexport const SearchInput = forwardRef(\n  (\n    {\n      placeholder,\n      label,\n      labelDescription,\n      loading,\n      size,\n      popupPlacement,\n      threshold = 0,\n      children,\n      onSearch,\n      onClose,\n      'data-testid': dataTestId,\n      shortcut = false,\n      error,\n      disabled,\n      className,\n      minLength,\n      maxLength,\n      tooltip,\n      onFocus,\n      onBlur,\n      name,\n      id,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      'aria-labelledby': ariaLabelledby,\n      readOnly,\n      required,\n      autoFocus,\n      autoComplete,\n      onKeyDown,\n      role,\n    }: SearchInputProps,\n    ref: Ref<HTMLInputElement>,\n  ) => {\n    const focusedLinkIndex = useRef(0)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const [containerWidth, setContainerWidth] = useState(0)\n    const [searchTerms, setSearchTerms] = useState('')\n    const [isMacOS, setIsMacOS] = useState(false)\n    const [keyPressed, setKeyPressed] = useState<string[]>([])\n    const [isOpen, toggleIsOpen] = useReducer(state => !state, false)\n    const innerSearchInputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(\n      ref,\n      () => innerSearchInputRef.current as HTMLInputElement,\n    )\n\n    const content =\n      typeof children === 'function'\n        ? children({ searchTerms, isOpen, toggleIsOpen })\n        : children\n\n    const resizeSearchBar = () => {\n      if (popupRef.current) {\n        setContainerWidth(popupRef.current.getBoundingClientRect().width)\n      }\n    }\n\n    const handleNavigation = (event: KeyboardEvent) => {\n      const links = [...(popupRef.current?.querySelectorAll('a') ?? [])]\n\n      if (\n        links.length > 0 &&\n        focusedLinkIndex.current >= 0 &&\n        focusedLinkIndex.current <= links.length\n      ) {\n        if (event.key === 'ArrowUp') {\n          if (focusedLinkIndex.current - 1 < 0) {\n            focusedLinkIndex.current = links.length - 1\n          } else {\n            focusedLinkIndex.current -= 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n\n        if (event.key === 'ArrowDown') {\n          if (focusedLinkIndex.current + 1 >= links.length) {\n            focusedLinkIndex.current = 0\n          } else {\n            focusedLinkIndex.current += 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n      }\n    }\n\n    useEffect(() => {\n      document.addEventListener('keyup', handleNavigation)\n\n      return () => document.removeEventListener('keyup', handleNavigation)\n    }, [])\n\n    useEffect(() => {\n      resizeSearchBar()\n\n      window.addEventListener('resize', resizeSearchBar)\n\n      return () => window.removeEventListener('resize', resizeSearchBar)\n    }, [])\n\n    const onSearchCallback = (localValue: string) => {\n      setSearchTerms(localValue)\n\n      try {\n        onSearch(localValue)\n        if (localValue.length >= threshold && !isOpen) {\n          toggleIsOpen()\n        }\n      } catch {\n        toggleIsOpen()\n      }\n    }\n\n    const onCloseCallback = () => {\n      onClose?.()\n      if (isOpen) {\n        toggleIsOpen()\n      }\n    }\n\n    useEffect(() => {\n      if (isClientSide) {\n        // We need to check if window is defined to avoid SSR issues\n        setIsMacOS(navigator.userAgent.includes('Mac'))\n      }\n    }, [])\n\n    const handleKeyPressed = useCallback(\n      (event: KeyboardEvent) => {\n        const { ctrlKey, metaKey, key } = event\n        setKeyPressed([...keyPressed, key.toUpperCase()])\n\n        if (typeof shortcut === 'boolean') {\n          if (\n            (key === 'k' || key === 'K') &&\n            ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        } else {\n          const uppercaseShortcut = shortcut.map(s => s.toUpperCase())\n\n          if (\n            JSON.stringify([...keyPressed, key.toUpperCase()]) ===\n            JSON.stringify(uppercaseShortcut)\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        }\n      },\n      [keyPressed, shortcut, isMacOS],\n    )\n\n    const handleKeyReleased = useCallback(\n      (event: KeyboardEvent) => {\n        const { key } = event\n        setKeyPressed(keyPressed.filter(k => k !== key.toUpperCase()))\n      },\n      [keyPressed],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleKeyPressed)\n        document.body.addEventListener('keyup', handleKeyReleased)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleKeyPressed)\n        document.body.removeEventListener('keyup', handleKeyReleased)\n      }\n    }, [shortcut, disabled, handleKeyPressed, handleKeyReleased])\n\n    const keys = useMemo(() => {\n      if (typeof shortcut === 'boolean') {\n        return [isMacOS ? '⌘' : 'Ctrl', 'K']\n      }\n\n      const filteredKey = shortcut.map(key => {\n        if (key === 'Meta') {\n          return '⌘'\n        }\n\n        if (key === 'Control') {\n          return 'Ctrl'\n        }\n\n        return key\n      })\n\n      return filteredKey\n    }, [isMacOS, shortcut])\n\n    return (\n      <div style={{ width: '100%' }}>\n        <StyledPopup\n          data-testid={`popup-${dataTestId}`}\n          role=\"dialog\"\n          visible={isOpen}\n          onClose={onCloseCallback}\n          placement={popupPlacement}\n          maxWidth={containerWidth}\n          hideOnClickOutside\n          hasArrow={false}\n          innerRef={popupRef}\n          text={content}\n          maxHeight={410}\n          debounceDelay={0}\n        >\n          <StyledTextInputV2\n            ref={innerSearchInputRef}\n            prefix={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup disabled={disabled} keys={keys} />\n              ) : undefined\n            }\n            data-testid={dataTestId}\n            error={error}\n            value={searchTerms}\n            size={size}\n            label={label}\n            placeholder={placeholder}\n            loading={loading}\n            onChange={onSearchCallback}\n            clearable\n            disabled={disabled}\n            className={className}\n            aria-atomic={ariaAtomic}\n            autoComplete={autoComplete}\n            aria-labelledby={ariaLabelledby}\n            aria-live={ariaLive}\n            id={id}\n            name={name}\n            readOnly={readOnly}\n            required={required}\n            autoFocus={autoFocus}\n            maxLength={maxLength}\n            minLength={minLength}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            onKeyDown={onKeyDown}\n            role={role}\n            tooltip={tooltip}\n            labelDescription={labelDescription}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
|
|
26
|
+
})(BasicPrefixStack, "{border:none;}", StyledInput, "{padding:0;}", BasicSuffixStack, "{border:none;}" + (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/SearchInput/index.tsx"],"names":[],"mappings":"AAiC6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\nimport { isClientSide } from '../../helpers/isClientSide'\nimport { Popup } from '../Popup'\nimport {\n  BasicPrefixStack,\n  BasicSuffixStack,\n  StyledInput,\n  TextInputV2,\n} from '../TextInputV2'\nimport { KeyGroup } from './KeyGroup'\nimport type { SearchInputProps } from './types'\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  text-align: initial;\n  min-width: 610px;\n  padding: ${({ theme }) => `${theme.space['2']} ${theme.space['1']}`};\n  background: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n`\n\nconst StyledTextInputV2 = styled(TextInputV2)`\n  ${BasicPrefixStack} {\n    border: none;\n  }\n\n  ${StyledInput} {\n    padding: 0;\n  }\n\n  ${BasicSuffixStack} {\n    border: none;\n  }\n`\n\n/**\n * SearchInput is a component that allows users to search for items. It is a combination of a TextInputV2 and a Popup. The Popup is used to display search results.\n * Children of the SearchInput component can be a function that receives an object with the following properties:\n * - `searchTerms`: the current search terms\n * - `isOpen`: a boolean indicating if the popup is open\n * - `toggleIsOpen`: a function to toggle the popup\n */\nexport const SearchInput = forwardRef(\n  (\n    {\n      placeholder,\n      label,\n      labelDescription,\n      loading,\n      size,\n      popupPlacement,\n      threshold = 0,\n      children,\n      onSearch,\n      onClose,\n      'data-testid': dataTestId,\n      shortcut = false,\n      error,\n      disabled,\n      className,\n      minLength,\n      maxLength,\n      tooltip,\n      onFocus,\n      onBlur,\n      name,\n      id,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      'aria-labelledby': ariaLabelledby,\n      readOnly,\n      required,\n      autoFocus,\n      autoComplete,\n      onKeyDown,\n      role,\n    }: SearchInputProps,\n    ref: Ref<HTMLInputElement>,\n  ) => {\n    const focusedLinkIndex = useRef(0)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const [containerWidth, setContainerWidth] = useState(0)\n    const [searchTerms, setSearchTerms] = useState('')\n    const [isMacOS, setIsMacOS] = useState(false)\n    const [keyPressed, setKeyPressed] = useState<string[]>([])\n    const [isOpen, toggleIsOpen] = useReducer(state => !state, false)\n    const innerSearchInputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(\n      ref,\n      () => innerSearchInputRef.current as HTMLInputElement,\n    )\n\n    const content =\n      typeof children === 'function'\n        ? children({ searchTerms, isOpen, toggleIsOpen })\n        : children\n\n    const resizeSearchBar = () => {\n      if (popupRef.current) {\n        setContainerWidth(popupRef.current.getBoundingClientRect().width)\n      }\n    }\n\n    const handleNavigation = (event: KeyboardEvent) => {\n      const links = [...(popupRef.current?.querySelectorAll('a') ?? [])]\n\n      if (\n        links.length > 0 &&\n        focusedLinkIndex.current >= 0 &&\n        focusedLinkIndex.current <= links.length\n      ) {\n        if (event.key === 'ArrowUp') {\n          if (focusedLinkIndex.current - 1 < 0) {\n            focusedLinkIndex.current = links.length - 1\n          } else {\n            focusedLinkIndex.current -= 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n\n        if (event.key === 'ArrowDown') {\n          if (focusedLinkIndex.current + 1 >= links.length) {\n            focusedLinkIndex.current = 0\n          } else {\n            focusedLinkIndex.current += 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n      }\n    }\n\n    useEffect(() => {\n      document.addEventListener('keyup', handleNavigation)\n\n      return () => document.removeEventListener('keyup', handleNavigation)\n    }, [])\n\n    useEffect(() => {\n      resizeSearchBar()\n\n      window.addEventListener('resize', resizeSearchBar)\n\n      return () => window.removeEventListener('resize', resizeSearchBar)\n    }, [])\n\n    const onSearchCallback = (localValue: string) => {\n      setSearchTerms(localValue)\n\n      try {\n        onSearch(localValue)\n        if (localValue.length >= threshold && !isOpen) {\n          toggleIsOpen()\n        }\n      } catch {\n        toggleIsOpen()\n      }\n    }\n\n    const onCloseCallback = () => {\n      onClose?.()\n      if (isOpen) {\n        toggleIsOpen()\n      }\n    }\n\n    useEffect(() => {\n      if (isClientSide) {\n        // We need to check if window is defined to avoid SSR issues\n        setIsMacOS(navigator.userAgent.includes('Mac'))\n      }\n    }, [])\n\n    const handleKeyPressed = useCallback(\n      (event: KeyboardEvent) => {\n        if (!(event instanceof KeyboardEvent)) {\n          return\n        }\n\n        const { ctrlKey, metaKey, key } = event\n        setKeyPressed([...keyPressed, key.toUpperCase()])\n\n        if (typeof shortcut === 'boolean') {\n          if (\n            (key === 'k' || key === 'K') &&\n            ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        } else {\n          const uppercaseShortcut = shortcut.map(s => s.toUpperCase())\n\n          if (\n            JSON.stringify([...keyPressed, key.toUpperCase()]) ===\n            JSON.stringify(uppercaseShortcut)\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        }\n      },\n      [keyPressed, shortcut, isMacOS],\n    )\n\n    const handleKeyReleased = useCallback(\n      (event: KeyboardEvent) => {\n        const { key } = event\n        setKeyPressed(keyPressed.filter(k => k !== key.toUpperCase()))\n      },\n      [keyPressed],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleKeyPressed)\n        document.body.addEventListener('keyup', handleKeyReleased)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleKeyPressed)\n        document.body.removeEventListener('keyup', handleKeyReleased)\n      }\n    }, [shortcut, disabled, handleKeyPressed, handleKeyReleased])\n\n    const keys = useMemo(() => {\n      if (typeof shortcut === 'boolean') {\n        return [isMacOS ? '⌘' : 'Ctrl', 'K']\n      }\n\n      const filteredKey = shortcut.map(key => {\n        if (key === 'Meta') {\n          return '⌘'\n        }\n\n        if (key === 'Control') {\n          return 'Ctrl'\n        }\n\n        return key\n      })\n\n      return filteredKey\n    }, [isMacOS, shortcut])\n\n    return (\n      <div style={{ width: '100%' }}>\n        <StyledPopup\n          data-testid={`popup-${dataTestId}`}\n          role=\"dialog\"\n          visible={isOpen}\n          onClose={onCloseCallback}\n          placement={popupPlacement}\n          maxWidth={containerWidth}\n          hideOnClickOutside\n          hasArrow={false}\n          innerRef={popupRef}\n          text={content}\n          maxHeight={410}\n          debounceDelay={0}\n        >\n          <StyledTextInputV2\n            ref={innerSearchInputRef}\n            prefix={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup disabled={disabled} keys={keys} />\n              ) : undefined\n            }\n            data-testid={dataTestId}\n            error={error}\n            value={searchTerms}\n            size={size}\n            label={label}\n            placeholder={placeholder}\n            loading={loading}\n            onChange={onSearchCallback}\n            clearable\n            disabled={disabled}\n            className={className}\n            aria-atomic={ariaAtomic}\n            autoComplete={autoComplete}\n            aria-labelledby={ariaLabelledby}\n            aria-live={ariaLive}\n            id={id}\n            name={name}\n            readOnly={readOnly}\n            required={required}\n            autoFocus={autoFocus}\n            maxLength={maxLength}\n            minLength={minLength}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            onKeyDown={onKeyDown}\n            role={role}\n            tooltip={tooltip}\n            labelDescription={labelDescription}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
|
|
27
27
|
const SearchInput = forwardRef(({
|
|
28
28
|
placeholder,
|
|
29
29
|
label,
|
|
@@ -129,6 +129,9 @@ const SearchInput = forwardRef(({
|
|
|
129
129
|
}
|
|
130
130
|
}, []);
|
|
131
131
|
const handleKeyPressed = useCallback((event) => {
|
|
132
|
+
if (!(event instanceof KeyboardEvent)) {
|
|
133
|
+
return;
|
|
134
|
+
}
|
|
132
135
|
const {
|
|
133
136
|
ctrlKey,
|
|
134
137
|
metaKey,
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@ultraviolet/ui",
|
|
3
|
-
"version": "1.
|
|
3
|
+
"version": "1.61.0",
|
|
4
4
|
"description": "Ultraviolet UI",
|
|
5
5
|
"homepage": "https://github.com/scaleway/ultraviolet#readme",
|
|
6
6
|
"repository": {
|
|
@@ -68,7 +68,7 @@
|
|
|
68
68
|
"@types/react-dom": "18.3.0",
|
|
69
69
|
"react": "18.3.1",
|
|
70
70
|
"react-dom": "18.3.1",
|
|
71
|
-
"@ultraviolet/icons": "2.
|
|
71
|
+
"@ultraviolet/icons": "2.13.0",
|
|
72
72
|
"@ultraviolet/themes": "1.12.1",
|
|
73
73
|
"@utils/test": "0.0.1"
|
|
74
74
|
},
|
|
@@ -88,7 +88,7 @@
|
|
|
88
88
|
"react-use-clipboard": "1.0.9",
|
|
89
89
|
"reakit": "1.3.11",
|
|
90
90
|
"@ultraviolet/themes": "1.12.1",
|
|
91
|
-
"@ultraviolet/icons": "2.
|
|
91
|
+
"@ultraviolet/icons": "2.13.0"
|
|
92
92
|
},
|
|
93
93
|
"scripts": {
|
|
94
94
|
"prebuild": "shx rm -rf dist",
|