@ultraviolet/ui 1.82.2 → 1.82.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/components/Badge/index.cjs +3 -3
- package/dist/components/Badge/index.js +3 -3
- package/dist/components/Modal/components/Dialog.cjs +4 -7
- package/dist/components/Modal/components/Dialog.js +4 -7
- package/dist/components/Slider/components/DoubleSlider.cjs +13 -10
- package/dist/components/Slider/components/DoubleSlider.js +13 -10
- package/dist/components/Snippet/index.cjs +13 -15
- package/dist/components/Snippet/index.js +13 -15
- package/dist/components/Stepper/Step.cjs +8 -8
- package/dist/components/Stepper/Step.js +8 -8
- package/dist/components/Text/index.cjs +11 -5
- package/dist/components/Text/index.d.ts +6 -4
- package/dist/components/Text/index.js +11 -5
- package/package.json +8 -8
|
@@ -60,7 +60,7 @@ const StyledSpan = /* @__PURE__ */ _styled__default.default(index.Text, process.
|
|
|
60
60
|
shouldForwardProp: (prop) => !["sentimentStyles", "size", "fontSize"].includes(prop),
|
|
61
61
|
target: "ej33bna0",
|
|
62
62
|
label: "StyledSpan"
|
|
63
|
-
})("display:inline-flex;align-items:center;justify-content:center;
|
|
63
|
+
})("display:inline-flex;align-items:center;justify-content:center;border-radius:", ({
|
|
64
64
|
theme
|
|
65
65
|
}) => theme.radii.xlarge, ";padding:0 ", ({
|
|
66
66
|
theme,
|
|
@@ -77,7 +77,7 @@ const StyledSpan = /* @__PURE__ */ _styled__default.default(index.Text, process.
|
|
|
77
77
|
theme
|
|
78
78
|
}) => theme.sizing[SIZES[size]], ";text-transform:uppercase;", ({
|
|
79
79
|
sentimentStyles
|
|
80
|
-
}) => sentimentStyles, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL0JhZGdlL2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUErRkUiLCJmaWxlIjoiL2hvbWUvcnVubmVyL3dvcmsvdWx0cmF2aW9sZXQvdWx0cmF2aW9sZXQvcGFja2FnZXMvdWkvc3JjL2NvbXBvbmVudHMvQmFkZ2UvaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBUaGVtZSB9IGZyb20gJ0BlbW90aW9uL3JlYWN0J1xuaW1wb3J0IHsgdXNlVGhlbWUgfSBmcm9tICdAZW1vdGlvbi9yZWFjdCdcbmltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJ1xuaW1wb3J0IHsgSWNvbiB9IGZyb20gJ0B1bHRyYXZpb2xldC9pY29ucy9sZWdhY3knXG5pbXBvcnQgdHlwZSB7IENvbXBvbmVudFByb3BzLCBSZWFjdE5vZGUgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZU1lbW8gfSBmcm9tICdyZWFjdCdcbmltcG9ydCB0eXBlIHsgQ29sb3IgfSBmcm9tICcuLi8uLi90aGVtZSdcbmltcG9ydCB7IFNFTlRJTUVOVFNfV0lUSE9VVF9ORVVUUkFMIH0gZnJvbSAnLi4vLi4vdGhlbWUnXG5pbXBvcnQgY2FwaXRhbGl6ZSBmcm9tICcuLi8uLi91dGlscy9jYXBpdGFsaXplJ1xuaW1wb3J0IHsgVGV4dCB9IGZyb20gJy4uL1RleHQnXG5cbnR5cGUgSWNvbk5hbWUgPSBDb21wb25lbnRQcm9wczx0eXBlb2YgSWNvbj5bJ25hbWUnXVxuXG5leHBvcnQgY29uc3QgU0laRVMgPSB7XG4gIGxhcmdlOiAnNDAwJywgLy8gc2l6aW5nIHRva2VuIGZyb20gdGhlbWVcbiAgbWVkaXVtOiAnMzAwJyxcbiAgc21hbGw6ICcyMDAnLFxufSBhcyBjb25zdFxuXG5leHBvcnQgY29uc3QgUFJPTUlORU5DRVMgPSB7XG4gIGRlZmF1bHQ6ICdkZWZhdWx0JyxcbiAgc3Ryb25nOiAnc3Ryb25nJyxcbn1cblxuZXhwb3J0IGNvbnN0IFRFWFRfVkFSSUFOVCA9IHtcbiAgbGFyZ2U6ICdib2R5U21hbGwnLFxuICBtZWRpdW06ICdjYXB0aW9uJyxcbiAgc21hbGw6ICdjYXB0aW9uU21hbGwnLFxufSBhcyBjb25zdFxuXG4vKipcbiAqIEdlbmVyYXRlIGFsbCBzdHlsZXMgYXZhaWxhYmxlIGZvciBiYWRnZSBiYXNlZCBvbiBwcm9taW5lbmNlIGFuZCBzZW50aW1lbnRzXG4gKi9cbmNvbnN0IGdlbmVyYXRlU3R5bGVzID0gKHtcbiAgcHJvbWluZW5jZSxcbiAgdGhlbWUsXG59OiB7XG4gIHByb21pbmVuY2U6IHN0cmluZ1xuICB0aGVtZTogVGhlbWVcbn0pOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+ID0+IHtcbiAgY29uc3QgZGVmaW5lZFByb21pbmVuY2UgPVxuICAgIHByb21pbmVuY2UgPT09IFBST01JTkVOQ0VTLnN0cm9uZyA/IGNhcGl0YWxpemUoUFJPTUlORU5DRVMuc3Ryb25nKSA6ICcnXG5cbiAgY29uc3QgdGV4dCA9IGB0ZXh0JHtkZWZpbmVkUHJvbWluZW5jZX1gXG4gIGNvbnN0IGJhY2tncm91bmQgPSBgYmFja2dyb3VuZCR7ZGVmaW5lZFByb21pbmVuY2V9YFxuXG4gIHJldHVybiB7XG4gICAgLi4uU0VOVElNRU5UU19XSVRIT1VUX05FVVRSQUwucmVkdWNlKFxuICAgICAgKHJlZHVjZXIsIHNlbnRpbWVudCkgPT4gKHtcbiAgICAgICAgLi4ucmVkdWNlcixcbiAgICAgICAgW3NlbnRpbWVudF06IGBcbiAgICAgIGNvbG9yOiAke1xuICAgICAgICB0aGVtZS5jb2xvcnNbc2VudGltZW50XVt0ZXh0IGFzIGtleW9mIHR5cGVvZiB0aGVtZS5jb2xvcnMucHJpbWFyeV1cbiAgICAgIH07XG4gICAgICBiYWNrZ3JvdW5kOiAke1xuICAgICAgICB0aGVtZS5jb2xvcnNbc2VudGltZW50XVtiYWNrZ3JvdW5kIGFzIGtleW9mIHR5cGVvZiB0aGVtZS5jb2xvcnMucHJpbWFyeV1cbiAgICAgIH07XG4gICAgICBib3JkZXI6IDFweCBzb2xpZCAke1xuICAgICAgICB0aGVtZS5jb2xvcnNbc2VudGltZW50XVtiYWNrZ3JvdW5kIGFzIGtleW9mIHR5cGVvZiB0aGVtZS5jb2xvcnMucHJpbWFyeV1cbiAgICAgIH07XG4gICAgYCxcbiAgICAgIH0pLFxuICAgICAge30sXG4gICAgKSxcbiAgICBkaXNhYmxlZDogYFxuICAgICAgY29sb3I6ICR7dGhlbWUuY29sb3JzLm5ldXRyYWwudGV4dFdlYWt9O1xuICAgICAgYmFja2dyb3VuZDogJHt0aGVtZS5jb2xvcnMubmV1dHJhbC5iYWNrZ3JvdW5kU3Ryb25nfTtcbiAgICAgIGJvcmRlcjogbm9uZTtcbiAgICBgLFxuICAgIG5ldXRyYWw6IGBcbiAgICAgIGNvbG9yOiAke1xuICAgICAgICBwcm9taW5lbmNlID09PSBQUk9NSU5FTkNFUy5zdHJvbmdcbiAgICAgICAgICA/
|
|
80
|
+
}) => sentimentStyles, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL0JhZGdlL2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUErRkUiLCJmaWxlIjoiL2hvbWUvcnVubmVyL3dvcmsvdWx0cmF2aW9sZXQvdWx0cmF2aW9sZXQvcGFja2FnZXMvdWkvc3JjL2NvbXBvbmVudHMvQmFkZ2UvaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBUaGVtZSB9IGZyb20gJ0BlbW90aW9uL3JlYWN0J1xuaW1wb3J0IHsgdXNlVGhlbWUgfSBmcm9tICdAZW1vdGlvbi9yZWFjdCdcbmltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJ1xuaW1wb3J0IHsgSWNvbiB9IGZyb20gJ0B1bHRyYXZpb2xldC9pY29ucy9sZWdhY3knXG5pbXBvcnQgdHlwZSB7IENvbXBvbmVudFByb3BzLCBSZWFjdE5vZGUgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZU1lbW8gfSBmcm9tICdyZWFjdCdcbmltcG9ydCB0eXBlIHsgQ29sb3IgfSBmcm9tICcuLi8uLi90aGVtZSdcbmltcG9ydCB7IFNFTlRJTUVOVFNfV0lUSE9VVF9ORVVUUkFMIH0gZnJvbSAnLi4vLi4vdGhlbWUnXG5pbXBvcnQgY2FwaXRhbGl6ZSBmcm9tICcuLi8uLi91dGlscy9jYXBpdGFsaXplJ1xuaW1wb3J0IHsgVGV4dCB9IGZyb20gJy4uL1RleHQnXG5cbnR5cGUgSWNvbk5hbWUgPSBDb21wb25lbnRQcm9wczx0eXBlb2YgSWNvbj5bJ25hbWUnXVxuXG5leHBvcnQgY29uc3QgU0laRVMgPSB7XG4gIGxhcmdlOiAnNDAwJywgLy8gc2l6aW5nIHRva2VuIGZyb20gdGhlbWVcbiAgbWVkaXVtOiAnMzAwJyxcbiAgc21hbGw6ICcyMDAnLFxufSBhcyBjb25zdFxuXG5leHBvcnQgY29uc3QgUFJPTUlORU5DRVMgPSB7XG4gIGRlZmF1bHQ6ICdkZWZhdWx0JyxcbiAgc3Ryb25nOiAnc3Ryb25nJyxcbn1cblxuZXhwb3J0IGNvbnN0IFRFWFRfVkFSSUFOVCA9IHtcbiAgbGFyZ2U6ICdib2R5U21hbGwnLFxuICBtZWRpdW06ICdjYXB0aW9uJyxcbiAgc21hbGw6ICdjYXB0aW9uU21hbGwnLFxufSBhcyBjb25zdFxuXG4vKipcbiAqIEdlbmVyYXRlIGFsbCBzdHlsZXMgYXZhaWxhYmxlIGZvciBiYWRnZSBiYXNlZCBvbiBwcm9taW5lbmNlIGFuZCBzZW50aW1lbnRzXG4gKi9cbmNvbnN0IGdlbmVyYXRlU3R5bGVzID0gKHtcbiAgcHJvbWluZW5jZSxcbiAgdGhlbWUsXG59OiB7XG4gIHByb21pbmVuY2U6IHN0cmluZ1xuICB0aGVtZTogVGhlbWVcbn0pOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+ID0+IHtcbiAgY29uc3QgZGVmaW5lZFByb21pbmVuY2UgPVxuICAgIHByb21pbmVuY2UgPT09IFBST01JTkVOQ0VTLnN0cm9uZyA/IGNhcGl0YWxpemUoUFJPTUlORU5DRVMuc3Ryb25nKSA6ICcnXG5cbiAgY29uc3QgdGV4dCA9IGB0ZXh0JHtkZWZpbmVkUHJvbWluZW5jZX1gXG4gIGNvbnN0IGJhY2tncm91bmQgPSBgYmFja2dyb3VuZCR7ZGVmaW5lZFByb21pbmVuY2V9YFxuXG4gIHJldHVybiB7XG4gICAgLi4uU0VOVElNRU5UU19XSVRIT1VUX05FVVRSQUwucmVkdWNlKFxuICAgICAgKHJlZHVjZXIsIHNlbnRpbWVudCkgPT4gKHtcbiAgICAgICAgLi4ucmVkdWNlcixcbiAgICAgICAgW3NlbnRpbWVudF06IGBcbiAgICAgIGNvbG9yOiAke1xuICAgICAgICB0aGVtZS5jb2xvcnNbc2VudGltZW50XVt0ZXh0IGFzIGtleW9mIHR5cGVvZiB0aGVtZS5jb2xvcnMucHJpbWFyeV1cbiAgICAgIH07XG4gICAgICBiYWNrZ3JvdW5kOiAke1xuICAgICAgICB0aGVtZS5jb2xvcnNbc2VudGltZW50XVtiYWNrZ3JvdW5kIGFzIGtleW9mIHR5cGVvZiB0aGVtZS5jb2xvcnMucHJpbWFyeV1cbiAgICAgIH07XG4gICAgICBib3JkZXI6IDFweCBzb2xpZCAke1xuICAgICAgICB0aGVtZS5jb2xvcnNbc2VudGltZW50XVtiYWNrZ3JvdW5kIGFzIGtleW9mIHR5cGVvZiB0aGVtZS5jb2xvcnMucHJpbWFyeV1cbiAgICAgIH07XG4gICAgYCxcbiAgICAgIH0pLFxuICAgICAge30sXG4gICAgKSxcbiAgICBkaXNhYmxlZDogYFxuICAgICAgY29sb3I6ICR7dGhlbWUuY29sb3JzLm5ldXRyYWwudGV4dFdlYWt9O1xuICAgICAgYmFja2dyb3VuZDogJHt0aGVtZS5jb2xvcnMubmV1dHJhbC5iYWNrZ3JvdW5kU3Ryb25nfTtcbiAgICAgIGJvcmRlcjogbm9uZTtcbiAgICBgLFxuICAgIG5ldXRyYWw6IGBcbiAgICAgIGNvbG9yOiAke1xuICAgICAgICBwcm9taW5lbmNlID09PSBQUk9NSU5FTkNFUy5zdHJvbmdcbiAgICAgICAgICA/IHRoZW1lLmNvbG9ycy5uZXV0cmFsLnRleHRTdHJvbmdlclxuICAgICAgICAgIDogdGhlbWUuY29sb3JzLm5ldXRyYWwudGV4dFxuICAgICAgfTtcbiAgICAgIGJhY2tncm91bmQ6ICR7XG4gICAgICAgIHByb21pbmVuY2UgPT09IFBST01JTkVOQ0VTLnN0cm9uZ1xuICAgICAgICAgID8gdGhlbWUuY29sb3JzLm5ldXRyYWwuYmFja2dyb3VuZFN0cm9uZ2VyXG4gICAgICAgICAgOiB0aGVtZS5jb2xvcnMubmV1dHJhbC5iYWNrZ3JvdW5kV2Vha1xuICAgICAgfTtcbiAgICAgIGJvcmRlcjogMXB4IHNvbGlkICR7XG4gICAgICAgIHByb21pbmVuY2UgPT09IFBST01JTkVOQ0VTLnN0cm9uZ1xuICAgICAgICAgID8gdGhlbWUuY29sb3JzLm5ldXRyYWwuYm9yZGVyU3Ryb25nZXJcbiAgICAgICAgICA6IHRoZW1lLmNvbG9ycy5uZXV0cmFsLmJvcmRlclxuICAgICAgfTtcbiAgICBgLFxuICB9XG59XG5cbmNvbnN0IFN0eWxlZFNwYW4gPSBzdHlsZWQoVGV4dCwge1xuICBzaG91bGRGb3J3YXJkUHJvcDogcHJvcCA9PlxuICAgICFbJ3NlbnRpbWVudFN0eWxlcycsICdzaXplJywgJ2ZvbnRTaXplJ10uaW5jbHVkZXMocHJvcCksXG59KTx7XG4gIHNpemU6IGtleW9mIHR5cGVvZiBTSVpFU1xuICBzZW50aW1lbnRTdHlsZXM6IHN0cmluZ1xufT5gXG4gIGRpc3BsYXk6IGlubGluZS1mbGV4O1xuICBhbGlnbi1pdGVtczogY2VudGVyO1xuICBqdXN0aWZ5LWNvbnRlbnQ6IGNlbnRlcjtcbiAgYm9yZGVyLXJhZGl1czogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5yYWRpaS54bGFyZ2V9O1xuICBwYWRkaW5nOiAwXG4gICAgJHsoeyB0aGVtZSwgc2l6ZSB9KSA9PiB7XG4gICAgICBpZiAoc2l6ZSA9PT0gJ3NtYWxsJykgcmV0dXJuIHRoZW1lLnNwYWNlWycxJ11cbiAgICAgIGlmIChzaXplID09PSAnbGFyZ2UnKSByZXR1cm4gdGhlbWUuc3BhY2VbJzInXVxuXG4gICAgICByZXR1cm4gdGhlbWUuc3BhY2VbJzEuNSddXG4gICAgfX07XG4gIGdhcDogJHsoeyB0aGVtZSwgc2l6ZSB9KSA9PlxuICAgIHNpemUgPT09ICdzbWFsbCcgPyB0aGVtZS5zcGFjZVsnMC41J10gOiB0aGVtZS5zcGFjZVsnMSddfTtcbiAgd2lkdGg6IGZpdC1jb250ZW50O1xuICBoZWlnaHQ6ICR7KHsgc2l6ZSwgdGhlbWUgfSkgPT4gdGhlbWUuc2l6aW5nW1NJWkVTW3NpemVdXX07XG4gIHRleHQtdHJhbnNmb3JtOiB1cHBlcmNhc2U7XG4gICR7KHsgc2VudGltZW50U3R5bGVzIH0pID0+IHNlbnRpbWVudFN0eWxlc31cbmBcblxudHlwZSBCYWRnZVByb3BzID0ge1xuICBzZW50aW1lbnQ/OiBDb2xvclxuICBzaXplPzoga2V5b2YgdHlwZW9mIFNJWkVTXG4gIHByb21pbmVuY2U/OiBrZXlvZiB0eXBlb2YgUFJPTUlORU5DRVNcbiAgLyoqXG4gICAqIERlZmluZXMgaWNvbiB0byBkaXNwbGF5IG9uIGxlZnQgc2lkZSBvZiBiYWRnZS4gKipPbmx5IGF2YWlsYWJsZSBvbiBtZWRpdW0gYW5kIGxhcmdlIHNpemVzKiouXG4gICAqL1xuICBpY29uPzogSWNvbk5hbWVcbiAgZGlzYWJsZWQ/OiBib29sZWFuXG4gIGNsYXNzTmFtZT86IHN0cmluZ1xuICBjaGlsZHJlbjogUmVhY3ROb2RlXG4gICdkYXRhLXRlc3RpZCc/OiBzdHJpbmdcbn1cblxuLyoqXG4gKiBCYWRnZSBjb21wb25lbnQgaXMgdXNlZCB0byBkaXNwbGF5IGEgc3RhdHVzIG9yIGEgbGFiZWwgaW4gYSBzbWFsbCBjb250YWluZXIuXG4gKi9cbmV4cG9ydCBjb25zdCBCYWRnZSA9ICh7XG4gIHNlbnRpbWVudCA9ICduZXV0cmFsJyxcbiAgc2l6ZSA9ICdtZWRpdW0nLFxuICBwcm9taW5lbmNlID0gJ2RlZmF1bHQnLFxuICBpY29uLFxuICBkaXNhYmxlZCA9IGZhbHNlLFxuICBjbGFzc05hbWUsXG4gIGNoaWxkcmVuLFxuICAnZGF0YS10ZXN0aWQnOiBkYXRhVGVzdElkLFxufTogQmFkZ2VQcm9wcykgPT4ge1xuICBjb25zdCB0aGVtZSA9IHVzZVRoZW1lKClcblxuICAvKipcbiAgICogQmFkZ2Ugc2hvdWxkIGRpc3BsYXkgYW4gYXJpYS1sYWJlbCBpZiB0aGUgc3RhdHVzIGlzIG5vdCBuZXV0cmFsIG9yIHByaW1hcnlcbiAgICovXG4gIGNvbnN0IGFyaWFMYWJlbCA9IHVzZU1lbW8oXG4gICAgKCkgPT5cbiAgICAgIFsnbmV1dHJhbCcsICdwcmltYXJ5J10uc29tZShiYXNlU2VudGltZW50ID0+IGJhc2VTZW50aW1lbnQgPT09IHNlbnRpbWVudClcbiAgICAgICAgPyB1bmRlZmluZWRcbiAgICAgICAgOiBzZW50aW1lbnQsXG4gICAgW3NlbnRpbWVudF0sXG4gIClcblxuICBjb25zdCBnZW5lcmF0ZWRTdHlsZXMgPSB1c2VNZW1vKFxuICAgICgpID0+IGdlbmVyYXRlU3R5bGVzKHsgcHJvbWluZW5jZSwgdGhlbWUgfSksXG4gICAgW3Byb21pbmVuY2UsIHRoZW1lXSxcbiAgKVxuXG4gIHJldHVybiAoXG4gICAgPFN0eWxlZFNwYW5cbiAgICAgIGFyaWEtbGFiZWw9e2FyaWFMYWJlbH1cbiAgICAgIHNlbnRpbWVudFN0eWxlcz17XG4gICAgICAgIGRpc2FibGVkID8gZ2VuZXJhdGVkU3R5bGVzWydkaXNhYmxlZCddIDogZ2VuZXJhdGVkU3R5bGVzW3NlbnRpbWVudF1cbiAgICAgIH1cbiAgICAgIHNpemU9e3NpemV9XG4gICAgICB2YXJpYW50PXtURVhUX1ZBUklBTlRbc2l6ZV19XG4gICAgICBhcz1cInNwYW5cIlxuICAgICAgcHJvbWluZW5jZT17ZGlzYWJsZWQgPyAnd2VhaycgOiAnZGVmYXVsdCd9XG4gICAgICBjbGFzc05hbWU9e2NsYXNzTmFtZX1cbiAgICAgIGRhdGEtdGVzdGlkPXtkYXRhVGVzdElkfVxuICAgICAgd2hpdGVTcGFjZT1cIm5vd3JhcFwiXG4gICAgPlxuICAgICAge2ljb24gJiYgc2l6ZSAhPT0gJ3NtYWxsJyA/IDxJY29uIG5hbWU9e2ljb259IHNpemU9ezE2fSAvPiA6IG51bGx9XG4gICAgICB7Y2hpbGRyZW59XG4gICAgPC9TdHlsZWRTcGFuPlxuICApXG59XG4iXX0= */"));
|
|
81
81
|
const Badge = ({
|
|
82
82
|
sentiment = "neutral",
|
|
83
83
|
size = "medium",
|
|
@@ -94,7 +94,7 @@ const Badge = ({
|
|
|
94
94
|
prominence,
|
|
95
95
|
theme
|
|
96
96
|
}), [prominence, theme]);
|
|
97
|
-
return /* @__PURE__ */ jsxRuntime.jsxs(StyledSpan, { "aria-label": ariaLabel, sentimentStyles: disabled ? generatedStyles["disabled"] : generatedStyles[sentiment], size, variant: TEXT_VARIANT[size], as: "span", prominence: disabled ? "weak" : "default", className, "data-testid": dataTestId, children: [
|
|
97
|
+
return /* @__PURE__ */ jsxRuntime.jsxs(StyledSpan, { "aria-label": ariaLabel, sentimentStyles: disabled ? generatedStyles["disabled"] : generatedStyles[sentiment], size, variant: TEXT_VARIANT[size], as: "span", prominence: disabled ? "weak" : "default", className, "data-testid": dataTestId, whiteSpace: "nowrap", children: [
|
|
98
98
|
icon && size !== "small" ? /* @__PURE__ */ jsxRuntime.jsx(legacy.Icon, { name: icon, size: 16 }) : null,
|
|
99
99
|
children
|
|
100
100
|
] });
|
|
@@ -56,7 +56,7 @@ const StyledSpan = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "produ
|
|
|
56
56
|
shouldForwardProp: (prop) => !["sentimentStyles", "size", "fontSize"].includes(prop),
|
|
57
57
|
target: "ej33bna0",
|
|
58
58
|
label: "StyledSpan"
|
|
59
|
-
})("display:inline-flex;align-items:center;justify-content:center;
|
|
59
|
+
})("display:inline-flex;align-items:center;justify-content:center;border-radius:", ({
|
|
60
60
|
theme
|
|
61
61
|
}) => theme.radii.xlarge, ";padding:0 ", ({
|
|
62
62
|
theme,
|
|
@@ -73,7 +73,7 @@ const StyledSpan = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "produ
|
|
|
73
73
|
theme
|
|
74
74
|
}) => theme.sizing[SIZES[size]], ";text-transform:uppercase;", ({
|
|
75
75
|
sentimentStyles
|
|
76
|
-
}) => sentimentStyles, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL0JhZGdlL2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUErRkUiLCJmaWxlIjoiL2hvbWUvcnVubmVyL3dvcmsvdWx0cmF2aW9sZXQvdWx0cmF2aW9sZXQvcGFja2FnZXMvdWkvc3JjL2NvbXBvbmVudHMvQmFkZ2UvaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBUaGVtZSB9IGZyb20gJ0BlbW90aW9uL3JlYWN0J1xuaW1wb3J0IHsgdXNlVGhlbWUgfSBmcm9tICdAZW1vdGlvbi9yZWFjdCdcbmltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJ1xuaW1wb3J0IHsgSWNvbiB9IGZyb20gJ0B1bHRyYXZpb2xldC9pY29ucy9sZWdhY3knXG5pbXBvcnQgdHlwZSB7IENvbXBvbmVudFByb3BzLCBSZWFjdE5vZGUgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZU1lbW8gfSBmcm9tICdyZWFjdCdcbmltcG9ydCB0eXBlIHsgQ29sb3IgfSBmcm9tICcuLi8uLi90aGVtZSdcbmltcG9ydCB7IFNFTlRJTUVOVFNfV0lUSE9VVF9ORVVUUkFMIH0gZnJvbSAnLi4vLi4vdGhlbWUnXG5pbXBvcnQgY2FwaXRhbGl6ZSBmcm9tICcuLi8uLi91dGlscy9jYXBpdGFsaXplJ1xuaW1wb3J0IHsgVGV4dCB9IGZyb20gJy4uL1RleHQnXG5cbnR5cGUgSWNvbk5hbWUgPSBDb21wb25lbnRQcm9wczx0eXBlb2YgSWNvbj5bJ25hbWUnXVxuXG5leHBvcnQgY29uc3QgU0laRVMgPSB7XG4gIGxhcmdlOiAnNDAwJywgLy8gc2l6aW5nIHRva2VuIGZyb20gdGhlbWVcbiAgbWVkaXVtOiAnMzAwJyxcbiAgc21hbGw6ICcyMDAnLFxufSBhcyBjb25zdFxuXG5leHBvcnQgY29uc3QgUFJPTUlORU5DRVMgPSB7XG4gIGRlZmF1bHQ6ICdkZWZhdWx0JyxcbiAgc3Ryb25nOiAnc3Ryb25nJyxcbn1cblxuZXhwb3J0IGNvbnN0IFRFWFRfVkFSSUFOVCA9IHtcbiAgbGFyZ2U6ICdib2R5U21hbGwnLFxuICBtZWRpdW06ICdjYXB0aW9uJyxcbiAgc21hbGw6ICdjYXB0aW9uU21hbGwnLFxufSBhcyBjb25zdFxuXG4vKipcbiAqIEdlbmVyYXRlIGFsbCBzdHlsZXMgYXZhaWxhYmxlIGZvciBiYWRnZSBiYXNlZCBvbiBwcm9taW5lbmNlIGFuZCBzZW50aW1lbnRzXG4gKi9cbmNvbnN0IGdlbmVyYXRlU3R5bGVzID0gKHtcbiAgcHJvbWluZW5jZSxcbiAgdGhlbWUsXG59OiB7XG4gIHByb21pbmVuY2U6IHN0cmluZ1xuICB0aGVtZTogVGhlbWVcbn0pOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+ID0+IHtcbiAgY29uc3QgZGVmaW5lZFByb21pbmVuY2UgPVxuICAgIHByb21pbmVuY2UgPT09IFBST01JTkVOQ0VTLnN0cm9uZyA/IGNhcGl0YWxpemUoUFJPTUlORU5DRVMuc3Ryb25nKSA6ICcnXG5cbiAgY29uc3QgdGV4dCA9IGB0ZXh0JHtkZWZpbmVkUHJvbWluZW5jZX1gXG4gIGNvbnN0IGJhY2tncm91bmQgPSBgYmFja2dyb3VuZCR7ZGVmaW5lZFByb21pbmVuY2V9YFxuXG4gIHJldHVybiB7XG4gICAgLi4uU0VOVElNRU5UU19XSVRIT1VUX05FVVRSQUwucmVkdWNlKFxuICAgICAgKHJlZHVjZXIsIHNlbnRpbWVudCkgPT4gKHtcbiAgICAgICAgLi4ucmVkdWNlcixcbiAgICAgICAgW3NlbnRpbWVudF06IGBcbiAgICAgIGNvbG9yOiAke1xuICAgICAgICB0aGVtZS5jb2xvcnNbc2VudGltZW50XVt0ZXh0IGFzIGtleW9mIHR5cGVvZiB0aGVtZS5jb2xvcnMucHJpbWFyeV1cbiAgICAgIH07XG4gICAgICBiYWNrZ3JvdW5kOiAke1xuICAgICAgICB0aGVtZS5jb2xvcnNbc2VudGltZW50XVtiYWNrZ3JvdW5kIGFzIGtleW9mIHR5cGVvZiB0aGVtZS5jb2xvcnMucHJpbWFyeV1cbiAgICAgIH07XG4gICAgICBib3JkZXI6IDFweCBzb2xpZCAke1xuICAgICAgICB0aGVtZS5jb2xvcnNbc2VudGltZW50XVtiYWNrZ3JvdW5kIGFzIGtleW9mIHR5cGVvZiB0aGVtZS5jb2xvcnMucHJpbWFyeV1cbiAgICAgIH07XG4gICAgYCxcbiAgICAgIH0pLFxuICAgICAge30sXG4gICAgKSxcbiAgICBkaXNhYmxlZDogYFxuICAgICAgY29sb3I6ICR7dGhlbWUuY29sb3JzLm5ldXRyYWwudGV4dFdlYWt9O1xuICAgICAgYmFja2dyb3VuZDogJHt0aGVtZS5jb2xvcnMubmV1dHJhbC5iYWNrZ3JvdW5kU3Ryb25nfTtcbiAgICAgIGJvcmRlcjogbm9uZTtcbiAgICBgLFxuICAgIG5ldXRyYWw6IGBcbiAgICAgIGNvbG9yOiAke1xuICAgICAgICBwcm9taW5lbmNlID09PSBQUk9NSU5FTkNFUy5zdHJvbmdcbiAgICAgICAgICA/
|
|
76
|
+
}) => sentimentStyles, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL0JhZGdlL2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUErRkUiLCJmaWxlIjoiL2hvbWUvcnVubmVyL3dvcmsvdWx0cmF2aW9sZXQvdWx0cmF2aW9sZXQvcGFja2FnZXMvdWkvc3JjL2NvbXBvbmVudHMvQmFkZ2UvaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBUaGVtZSB9IGZyb20gJ0BlbW90aW9uL3JlYWN0J1xuaW1wb3J0IHsgdXNlVGhlbWUgfSBmcm9tICdAZW1vdGlvbi9yZWFjdCdcbmltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJ1xuaW1wb3J0IHsgSWNvbiB9IGZyb20gJ0B1bHRyYXZpb2xldC9pY29ucy9sZWdhY3knXG5pbXBvcnQgdHlwZSB7IENvbXBvbmVudFByb3BzLCBSZWFjdE5vZGUgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZU1lbW8gfSBmcm9tICdyZWFjdCdcbmltcG9ydCB0eXBlIHsgQ29sb3IgfSBmcm9tICcuLi8uLi90aGVtZSdcbmltcG9ydCB7IFNFTlRJTUVOVFNfV0lUSE9VVF9ORVVUUkFMIH0gZnJvbSAnLi4vLi4vdGhlbWUnXG5pbXBvcnQgY2FwaXRhbGl6ZSBmcm9tICcuLi8uLi91dGlscy9jYXBpdGFsaXplJ1xuaW1wb3J0IHsgVGV4dCB9IGZyb20gJy4uL1RleHQnXG5cbnR5cGUgSWNvbk5hbWUgPSBDb21wb25lbnRQcm9wczx0eXBlb2YgSWNvbj5bJ25hbWUnXVxuXG5leHBvcnQgY29uc3QgU0laRVMgPSB7XG4gIGxhcmdlOiAnNDAwJywgLy8gc2l6aW5nIHRva2VuIGZyb20gdGhlbWVcbiAgbWVkaXVtOiAnMzAwJyxcbiAgc21hbGw6ICcyMDAnLFxufSBhcyBjb25zdFxuXG5leHBvcnQgY29uc3QgUFJPTUlORU5DRVMgPSB7XG4gIGRlZmF1bHQ6ICdkZWZhdWx0JyxcbiAgc3Ryb25nOiAnc3Ryb25nJyxcbn1cblxuZXhwb3J0IGNvbnN0IFRFWFRfVkFSSUFOVCA9IHtcbiAgbGFyZ2U6ICdib2R5U21hbGwnLFxuICBtZWRpdW06ICdjYXB0aW9uJyxcbiAgc21hbGw6ICdjYXB0aW9uU21hbGwnLFxufSBhcyBjb25zdFxuXG4vKipcbiAqIEdlbmVyYXRlIGFsbCBzdHlsZXMgYXZhaWxhYmxlIGZvciBiYWRnZSBiYXNlZCBvbiBwcm9taW5lbmNlIGFuZCBzZW50aW1lbnRzXG4gKi9cbmNvbnN0IGdlbmVyYXRlU3R5bGVzID0gKHtcbiAgcHJvbWluZW5jZSxcbiAgdGhlbWUsXG59OiB7XG4gIHByb21pbmVuY2U6IHN0cmluZ1xuICB0aGVtZTogVGhlbWVcbn0pOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+ID0+IHtcbiAgY29uc3QgZGVmaW5lZFByb21pbmVuY2UgPVxuICAgIHByb21pbmVuY2UgPT09IFBST01JTkVOQ0VTLnN0cm9uZyA/IGNhcGl0YWxpemUoUFJPTUlORU5DRVMuc3Ryb25nKSA6ICcnXG5cbiAgY29uc3QgdGV4dCA9IGB0ZXh0JHtkZWZpbmVkUHJvbWluZW5jZX1gXG4gIGNvbnN0IGJhY2tncm91bmQgPSBgYmFja2dyb3VuZCR7ZGVmaW5lZFByb21pbmVuY2V9YFxuXG4gIHJldHVybiB7XG4gICAgLi4uU0VOVElNRU5UU19XSVRIT1VUX05FVVRSQUwucmVkdWNlKFxuICAgICAgKHJlZHVjZXIsIHNlbnRpbWVudCkgPT4gKHtcbiAgICAgICAgLi4ucmVkdWNlcixcbiAgICAgICAgW3NlbnRpbWVudF06IGBcbiAgICAgIGNvbG9yOiAke1xuICAgICAgICB0aGVtZS5jb2xvcnNbc2VudGltZW50XVt0ZXh0IGFzIGtleW9mIHR5cGVvZiB0aGVtZS5jb2xvcnMucHJpbWFyeV1cbiAgICAgIH07XG4gICAgICBiYWNrZ3JvdW5kOiAke1xuICAgICAgICB0aGVtZS5jb2xvcnNbc2VudGltZW50XVtiYWNrZ3JvdW5kIGFzIGtleW9mIHR5cGVvZiB0aGVtZS5jb2xvcnMucHJpbWFyeV1cbiAgICAgIH07XG4gICAgICBib3JkZXI6IDFweCBzb2xpZCAke1xuICAgICAgICB0aGVtZS5jb2xvcnNbc2VudGltZW50XVtiYWNrZ3JvdW5kIGFzIGtleW9mIHR5cGVvZiB0aGVtZS5jb2xvcnMucHJpbWFyeV1cbiAgICAgIH07XG4gICAgYCxcbiAgICAgIH0pLFxuICAgICAge30sXG4gICAgKSxcbiAgICBkaXNhYmxlZDogYFxuICAgICAgY29sb3I6ICR7dGhlbWUuY29sb3JzLm5ldXRyYWwudGV4dFdlYWt9O1xuICAgICAgYmFja2dyb3VuZDogJHt0aGVtZS5jb2xvcnMubmV1dHJhbC5iYWNrZ3JvdW5kU3Ryb25nfTtcbiAgICAgIGJvcmRlcjogbm9uZTtcbiAgICBgLFxuICAgIG5ldXRyYWw6IGBcbiAgICAgIGNvbG9yOiAke1xuICAgICAgICBwcm9taW5lbmNlID09PSBQUk9NSU5FTkNFUy5zdHJvbmdcbiAgICAgICAgICA/IHRoZW1lLmNvbG9ycy5uZXV0cmFsLnRleHRTdHJvbmdlclxuICAgICAgICAgIDogdGhlbWUuY29sb3JzLm5ldXRyYWwudGV4dFxuICAgICAgfTtcbiAgICAgIGJhY2tncm91bmQ6ICR7XG4gICAgICAgIHByb21pbmVuY2UgPT09IFBST01JTkVOQ0VTLnN0cm9uZ1xuICAgICAgICAgID8gdGhlbWUuY29sb3JzLm5ldXRyYWwuYmFja2dyb3VuZFN0cm9uZ2VyXG4gICAgICAgICAgOiB0aGVtZS5jb2xvcnMubmV1dHJhbC5iYWNrZ3JvdW5kV2Vha1xuICAgICAgfTtcbiAgICAgIGJvcmRlcjogMXB4IHNvbGlkICR7XG4gICAgICAgIHByb21pbmVuY2UgPT09IFBST01JTkVOQ0VTLnN0cm9uZ1xuICAgICAgICAgID8gdGhlbWUuY29sb3JzLm5ldXRyYWwuYm9yZGVyU3Ryb25nZXJcbiAgICAgICAgICA6IHRoZW1lLmNvbG9ycy5uZXV0cmFsLmJvcmRlclxuICAgICAgfTtcbiAgICBgLFxuICB9XG59XG5cbmNvbnN0IFN0eWxlZFNwYW4gPSBzdHlsZWQoVGV4dCwge1xuICBzaG91bGRGb3J3YXJkUHJvcDogcHJvcCA9PlxuICAgICFbJ3NlbnRpbWVudFN0eWxlcycsICdzaXplJywgJ2ZvbnRTaXplJ10uaW5jbHVkZXMocHJvcCksXG59KTx7XG4gIHNpemU6IGtleW9mIHR5cGVvZiBTSVpFU1xuICBzZW50aW1lbnRTdHlsZXM6IHN0cmluZ1xufT5gXG4gIGRpc3BsYXk6IGlubGluZS1mbGV4O1xuICBhbGlnbi1pdGVtczogY2VudGVyO1xuICBqdXN0aWZ5LWNvbnRlbnQ6IGNlbnRlcjtcbiAgYm9yZGVyLXJhZGl1czogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5yYWRpaS54bGFyZ2V9O1xuICBwYWRkaW5nOiAwXG4gICAgJHsoeyB0aGVtZSwgc2l6ZSB9KSA9PiB7XG4gICAgICBpZiAoc2l6ZSA9PT0gJ3NtYWxsJykgcmV0dXJuIHRoZW1lLnNwYWNlWycxJ11cbiAgICAgIGlmIChzaXplID09PSAnbGFyZ2UnKSByZXR1cm4gdGhlbWUuc3BhY2VbJzInXVxuXG4gICAgICByZXR1cm4gdGhlbWUuc3BhY2VbJzEuNSddXG4gICAgfX07XG4gIGdhcDogJHsoeyB0aGVtZSwgc2l6ZSB9KSA9PlxuICAgIHNpemUgPT09ICdzbWFsbCcgPyB0aGVtZS5zcGFjZVsnMC41J10gOiB0aGVtZS5zcGFjZVsnMSddfTtcbiAgd2lkdGg6IGZpdC1jb250ZW50O1xuICBoZWlnaHQ6ICR7KHsgc2l6ZSwgdGhlbWUgfSkgPT4gdGhlbWUuc2l6aW5nW1NJWkVTW3NpemVdXX07XG4gIHRleHQtdHJhbnNmb3JtOiB1cHBlcmNhc2U7XG4gICR7KHsgc2VudGltZW50U3R5bGVzIH0pID0+IHNlbnRpbWVudFN0eWxlc31cbmBcblxudHlwZSBCYWRnZVByb3BzID0ge1xuICBzZW50aW1lbnQ/OiBDb2xvclxuICBzaXplPzoga2V5b2YgdHlwZW9mIFNJWkVTXG4gIHByb21pbmVuY2U/OiBrZXlvZiB0eXBlb2YgUFJPTUlORU5DRVNcbiAgLyoqXG4gICAqIERlZmluZXMgaWNvbiB0byBkaXNwbGF5IG9uIGxlZnQgc2lkZSBvZiBiYWRnZS4gKipPbmx5IGF2YWlsYWJsZSBvbiBtZWRpdW0gYW5kIGxhcmdlIHNpemVzKiouXG4gICAqL1xuICBpY29uPzogSWNvbk5hbWVcbiAgZGlzYWJsZWQ/OiBib29sZWFuXG4gIGNsYXNzTmFtZT86IHN0cmluZ1xuICBjaGlsZHJlbjogUmVhY3ROb2RlXG4gICdkYXRhLXRlc3RpZCc/OiBzdHJpbmdcbn1cblxuLyoqXG4gKiBCYWRnZSBjb21wb25lbnQgaXMgdXNlZCB0byBkaXNwbGF5IGEgc3RhdHVzIG9yIGEgbGFiZWwgaW4gYSBzbWFsbCBjb250YWluZXIuXG4gKi9cbmV4cG9ydCBjb25zdCBCYWRnZSA9ICh7XG4gIHNlbnRpbWVudCA9ICduZXV0cmFsJyxcbiAgc2l6ZSA9ICdtZWRpdW0nLFxuICBwcm9taW5lbmNlID0gJ2RlZmF1bHQnLFxuICBpY29uLFxuICBkaXNhYmxlZCA9IGZhbHNlLFxuICBjbGFzc05hbWUsXG4gIGNoaWxkcmVuLFxuICAnZGF0YS10ZXN0aWQnOiBkYXRhVGVzdElkLFxufTogQmFkZ2VQcm9wcykgPT4ge1xuICBjb25zdCB0aGVtZSA9IHVzZVRoZW1lKClcblxuICAvKipcbiAgICogQmFkZ2Ugc2hvdWxkIGRpc3BsYXkgYW4gYXJpYS1sYWJlbCBpZiB0aGUgc3RhdHVzIGlzIG5vdCBuZXV0cmFsIG9yIHByaW1hcnlcbiAgICovXG4gIGNvbnN0IGFyaWFMYWJlbCA9IHVzZU1lbW8oXG4gICAgKCkgPT5cbiAgICAgIFsnbmV1dHJhbCcsICdwcmltYXJ5J10uc29tZShiYXNlU2VudGltZW50ID0+IGJhc2VTZW50aW1lbnQgPT09IHNlbnRpbWVudClcbiAgICAgICAgPyB1bmRlZmluZWRcbiAgICAgICAgOiBzZW50aW1lbnQsXG4gICAgW3NlbnRpbWVudF0sXG4gIClcblxuICBjb25zdCBnZW5lcmF0ZWRTdHlsZXMgPSB1c2VNZW1vKFxuICAgICgpID0+IGdlbmVyYXRlU3R5bGVzKHsgcHJvbWluZW5jZSwgdGhlbWUgfSksXG4gICAgW3Byb21pbmVuY2UsIHRoZW1lXSxcbiAgKVxuXG4gIHJldHVybiAoXG4gICAgPFN0eWxlZFNwYW5cbiAgICAgIGFyaWEtbGFiZWw9e2FyaWFMYWJlbH1cbiAgICAgIHNlbnRpbWVudFN0eWxlcz17XG4gICAgICAgIGRpc2FibGVkID8gZ2VuZXJhdGVkU3R5bGVzWydkaXNhYmxlZCddIDogZ2VuZXJhdGVkU3R5bGVzW3NlbnRpbWVudF1cbiAgICAgIH1cbiAgICAgIHNpemU9e3NpemV9XG4gICAgICB2YXJpYW50PXtURVhUX1ZBUklBTlRbc2l6ZV19XG4gICAgICBhcz1cInNwYW5cIlxuICAgICAgcHJvbWluZW5jZT17ZGlzYWJsZWQgPyAnd2VhaycgOiAnZGVmYXVsdCd9XG4gICAgICBjbGFzc05hbWU9e2NsYXNzTmFtZX1cbiAgICAgIGRhdGEtdGVzdGlkPXtkYXRhVGVzdElkfVxuICAgICAgd2hpdGVTcGFjZT1cIm5vd3JhcFwiXG4gICAgPlxuICAgICAge2ljb24gJiYgc2l6ZSAhPT0gJ3NtYWxsJyA/IDxJY29uIG5hbWU9e2ljb259IHNpemU9ezE2fSAvPiA6IG51bGx9XG4gICAgICB7Y2hpbGRyZW59XG4gICAgPC9TdHlsZWRTcGFuPlxuICApXG59XG4iXX0= */"));
|
|
77
77
|
const Badge = ({
|
|
78
78
|
sentiment = "neutral",
|
|
79
79
|
size = "medium",
|
|
@@ -90,7 +90,7 @@ const Badge = ({
|
|
|
90
90
|
prominence,
|
|
91
91
|
theme
|
|
92
92
|
}), [prominence, theme]);
|
|
93
|
-
return /* @__PURE__ */ jsxs(StyledSpan, { "aria-label": ariaLabel, sentimentStyles: disabled ? generatedStyles["disabled"] : generatedStyles[sentiment], size, variant: TEXT_VARIANT[size], as: "span", prominence: disabled ? "weak" : "default", className, "data-testid": dataTestId, children: [
|
|
93
|
+
return /* @__PURE__ */ jsxs(StyledSpan, { "aria-label": ariaLabel, sentimentStyles: disabled ? generatedStyles["disabled"] : generatedStyles[sentiment], size, variant: TEXT_VARIANT[size], as: "span", prominence: disabled ? "weak" : "default", className, "data-testid": dataTestId, whiteSpace: "nowrap", children: [
|
|
94
94
|
icon && size !== "small" ? /* @__PURE__ */ jsx(Icon, { name: icon, size: 16 }) : null,
|
|
95
95
|
children
|
|
96
96
|
] });
|
|
@@ -18,7 +18,7 @@ const StyledBackdrop = /* @__PURE__ */ _styled__default.default("div", process.e
|
|
|
18
18
|
theme
|
|
19
19
|
}) => theme.colors.overlay, ";z-index:1;opacity:0;&[data-open='true']{padding:", ({
|
|
20
20
|
theme
|
|
21
|
-
}) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}&[data-visible='true']{opacity:1;}&[data-animation='true']{overflow:hidden;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx"],"names":[],"mappings":"AAc2D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  opacity: 0;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n\n  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n  &[data-animation='true'] {\n    overflow: hidden;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].includes(prop),\n})<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}rem;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\n}\n\nexport const Dialog = ({\n  children,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n  const {\n    registerModal,\n    unregisterModal,\n    openedModals,\n    previsousOpenedModales,\n  } = useModal()\n\n  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\n\n    return () => {\n      unregisterModal(id)\n    }\n  }, [id, registerModal, unregisterModal])\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    document.body.appendChild(element)\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [])\n\n  // Save the reassignment of eventHandler in the useEffect below\n  useEffect(() => {\n    onCloseRef.current = onClose\n  }, [onClose])\n\n  // On open focus the modal\n  useEffect(() => {\n    dialogRef.current?.focus()\n  }, [])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // Stop click to prevent unexpected dialog close\n  const stopClick: MouseEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (hideOnClickOutside) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // We need to reverse the array as the last opened modal should be the first to be with normal size\n  // while the first opened modal should shrink\n  const realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\n\n  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      css={backdropCss}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\n    >\n      <StyledDialog\n        css={dialogCss}\n        onKeyUp={handleKeyUp}\n        onKeyDown={handleFocusTrap}\n        className={className}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n        data-placement={placement}\n        data-size={size}\n        open\n        onClick={stopClick}\n        onCancel={stopCancel}\n        onClose={stopCancel}\n        onMouseDown={stopClick}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
21
|
+
}) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}&[data-visible='true']{opacity:1;}&[data-animation='true']{overflow:hidden;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx"],"names":[],"mappings":"AAc2D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  opacity: 0;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n\n  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n  &[data-animation='true'] {\n    overflow: hidden;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].includes(prop),\n})<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}rem;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\n}\n\nexport const Dialog = ({\n  children,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n  const {\n    registerModal,\n    unregisterModal,\n    openedModals,\n    previsousOpenedModales,\n  } = useModal()\n\n  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\n\n    return () => {\n      unregisterModal(id)\n    }\n  }, [id, registerModal, unregisterModal])\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    document.body.appendChild(element)\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [])\n\n  // Save the reassignment of eventHandler in the useEffect below\n  useEffect(() => {\n    onCloseRef.current = onClose\n  }, [onClose])\n\n  // On open focus the modal\n  useEffect(() => {\n    dialogRef.current?.focus()\n  }, [])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n\n      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node)\n      ) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // We need to reverse the array as the last opened modal should be the first to be with normal size\n  // while the first opened modal should shrink\n  const realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\n\n  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      css={backdropCss}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\n    >\n      <StyledDialog\n        css={dialogCss}\n        onKeyUp={handleKeyUp}\n        onKeyDown={handleFocusTrap}\n        className={className}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n        data-placement={placement}\n        data-size={size}\n        open\n        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
22
22
|
const StyledDialog = /* @__PURE__ */ _styled__default.default("dialog", process.env.NODE_ENV === "production" ? {
|
|
23
23
|
shouldForwardProp: (prop) => !["position", "size", "openedModals", "top"].includes(prop),
|
|
24
24
|
target: "e1fims040"
|
|
@@ -49,7 +49,7 @@ const StyledDialog = /* @__PURE__ */ _styled__default.default("dialog", process.
|
|
|
49
49
|
}) => position > 0 ? `
|
|
50
50
|
width: calc(${constants.MODAL_WIDTH[size]}rem - ${position * 50}px) !important;
|
|
51
51
|
transform: translate3d(0, -${top}px, 0);
|
|
52
|
-
` : void 0, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx"],"names":[],"mappings":"AAuDqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  opacity: 0;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n\n  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n  &[data-animation='true'] {\n    overflow: hidden;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].includes(prop),\n})<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}rem;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\n}\n\nexport const Dialog = ({\n  children,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n  const {\n    registerModal,\n    unregisterModal,\n    openedModals,\n    previsousOpenedModales,\n  } = useModal()\n\n  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\n\n    return () => {\n      unregisterModal(id)\n    }\n  }, [id, registerModal, unregisterModal])\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    document.body.appendChild(element)\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [])\n\n  // Save the reassignment of eventHandler in the useEffect below\n  useEffect(() => {\n    onCloseRef.current = onClose\n  }, [onClose])\n\n  // On open focus the modal\n  useEffect(() => {\n    dialogRef.current?.focus()\n  }, [])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // Stop click to prevent unexpected dialog close\n  const stopClick: MouseEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (hideOnClickOutside) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // We need to reverse the array as the last opened modal should be the first to be with normal size\n  // while the first opened modal should shrink\n  const realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\n\n  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      css={backdropCss}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\n    >\n      <StyledDialog\n        css={dialogCss}\n        onKeyUp={handleKeyUp}\n        onKeyDown={handleFocusTrap}\n        className={className}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n        data-placement={placement}\n        data-size={size}\n        open\n        onClick={stopClick}\n        onCancel={stopCancel}\n        onClose={stopCancel}\n        onMouseDown={stopClick}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
52
|
+
` : void 0, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx"],"names":[],"mappings":"AAuDqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  opacity: 0;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n\n  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n  &[data-animation='true'] {\n    overflow: hidden;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].includes(prop),\n})<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}rem;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\n}\n\nexport const Dialog = ({\n  children,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n  const {\n    registerModal,\n    unregisterModal,\n    openedModals,\n    previsousOpenedModales,\n  } = useModal()\n\n  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\n\n    return () => {\n      unregisterModal(id)\n    }\n  }, [id, registerModal, unregisterModal])\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    document.body.appendChild(element)\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [])\n\n  // Save the reassignment of eventHandler in the useEffect below\n  useEffect(() => {\n    onCloseRef.current = onClose\n  }, [onClose])\n\n  // On open focus the modal\n  useEffect(() => {\n    dialogRef.current?.focus()\n  }, [])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n\n      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node)\n      ) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // We need to reverse the array as the last opened modal should be the first to be with normal size\n  // while the first opened modal should shrink\n  const realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\n\n  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      css={backdropCss}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\n    >\n      <StyledDialog\n        css={dialogCss}\n        onKeyUp={handleKeyUp}\n        onKeyDown={handleFocusTrap}\n        className={className}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n        data-placement={placement}\n        data-size={size}\n        open\n        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
53
53
|
const stopCancel = (event) => {
|
|
54
54
|
event.preventDefault();
|
|
55
55
|
event.stopPropagation();
|
|
@@ -119,9 +119,6 @@ const Dialog = ({
|
|
|
119
119
|
const stopFocus = React.useCallback((event) => {
|
|
120
120
|
event.stopPropagation();
|
|
121
121
|
}, []);
|
|
122
|
-
const stopClick = React.useCallback((event) => {
|
|
123
|
-
event.stopPropagation();
|
|
124
|
-
}, []);
|
|
125
122
|
const handleKeyUp = React.useCallback((event) => {
|
|
126
123
|
event.stopPropagation();
|
|
127
124
|
if (event.key === "Escape" && hideOnEsc) {
|
|
@@ -131,7 +128,7 @@ const Dialog = ({
|
|
|
131
128
|
}, [hideOnEsc]);
|
|
132
129
|
const handleClose = React.useCallback((event) => {
|
|
133
130
|
event.stopPropagation();
|
|
134
|
-
if (hideOnClickOutside) {
|
|
131
|
+
if (hideOnClickOutside && dialogRef.current && !dialogRef.current.contains(event.target)) {
|
|
135
132
|
onCloseRef.current();
|
|
136
133
|
} else {
|
|
137
134
|
dialogRef.current?.focus();
|
|
@@ -172,7 +169,7 @@ const Dialog = ({
|
|
|
172
169
|
top = (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 - currentModalHeight / 2 + 20;
|
|
173
170
|
}
|
|
174
171
|
const animation = openedModals.length > 1 && position === 0 && previsousOpenedModales.length < openedModals.length;
|
|
175
|
-
return reactDom.createPortal(/* @__PURE__ */ jsxRuntime.jsx(StyledBackdrop, { "data-open": true, onClick: handleClose, className: backdropClassName, css: backdropCss, "data-testid": dataTestId ? `${dataTestId}-backdrop` : void 0, onFocus: stopFocus, "data-animation": animation, "data-visible": isVisible, id: "backdrop-modal", children: /* @__PURE__ */ jsxRuntime.jsx(StyledDialog, { css: dialogCss, onKeyUp: handleKeyUp, onKeyDown: handleFocusTrap, className, id, "data-testid": dataTestId, "aria-label": ariaLabel, "data-placement": placement, "data-size": size, open: true,
|
|
172
|
+
return reactDom.createPortal(/* @__PURE__ */ jsxRuntime.jsx(StyledBackdrop, { "data-open": true, onClick: handleClose, className: backdropClassName, css: backdropCss, "data-testid": dataTestId ? `${dataTestId}-backdrop` : void 0, onFocus: stopFocus, "data-animation": animation, "data-visible": isVisible, id: "backdrop-modal", children: /* @__PURE__ */ jsxRuntime.jsx(StyledDialog, { css: dialogCss, onKeyUp: handleKeyUp, onKeyDown: handleFocusTrap, className, id, "data-testid": dataTestId, "aria-label": ariaLabel, "data-placement": placement, "data-size": size, open: true, onCancel: stopCancel, onClose: stopCancel, "aria-modal": true, ref: dialogRef, tabIndex: 0, position, top: Math.max(top, 0), "data-animation": animation, size, children }) }), containerRef.current);
|
|
176
173
|
};
|
|
177
174
|
exports.Dialog = Dialog;
|
|
178
175
|
exports.StyledDialog = StyledDialog;
|
|
@@ -14,7 +14,7 @@ const StyledBackdrop = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "
|
|
|
14
14
|
theme
|
|
15
15
|
}) => theme.colors.overlay, ";z-index:1;opacity:0;&[data-open='true']{padding:", ({
|
|
16
16
|
theme
|
|
17
|
-
}) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}&[data-visible='true']{opacity:1;}&[data-animation='true']{overflow:hidden;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx"],"names":[],"mappings":"AAc2D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  opacity: 0;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n\n  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n  &[data-animation='true'] {\n    overflow: hidden;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].includes(prop),\n})<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}rem;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\n}\n\nexport const Dialog = ({\n  children,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n  const {\n    registerModal,\n    unregisterModal,\n    openedModals,\n    previsousOpenedModales,\n  } = useModal()\n\n  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\n\n    return () => {\n      unregisterModal(id)\n    }\n  }, [id, registerModal, unregisterModal])\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    document.body.appendChild(element)\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [])\n\n  // Save the reassignment of eventHandler in the useEffect below\n  useEffect(() => {\n    onCloseRef.current = onClose\n  }, [onClose])\n\n  // On open focus the modal\n  useEffect(() => {\n    dialogRef.current?.focus()\n  }, [])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // Stop click to prevent unexpected dialog close\n  const stopClick: MouseEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (hideOnClickOutside) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // We need to reverse the array as the last opened modal should be the first to be with normal size\n  // while the first opened modal should shrink\n  const realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\n\n  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      css={backdropCss}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\n    >\n      <StyledDialog\n        css={dialogCss}\n        onKeyUp={handleKeyUp}\n        onKeyDown={handleFocusTrap}\n        className={className}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n        data-placement={placement}\n        data-size={size}\n        open\n        onClick={stopClick}\n        onCancel={stopCancel}\n        onClose={stopCancel}\n        onMouseDown={stopClick}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
17
|
+
}) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}&[data-visible='true']{opacity:1;}&[data-animation='true']{overflow:hidden;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx"],"names":[],"mappings":"AAc2D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  opacity: 0;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n\n  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n  &[data-animation='true'] {\n    overflow: hidden;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].includes(prop),\n})<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}rem;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\n}\n\nexport const Dialog = ({\n  children,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n  const {\n    registerModal,\n    unregisterModal,\n    openedModals,\n    previsousOpenedModales,\n  } = useModal()\n\n  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\n\n    return () => {\n      unregisterModal(id)\n    }\n  }, [id, registerModal, unregisterModal])\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    document.body.appendChild(element)\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [])\n\n  // Save the reassignment of eventHandler in the useEffect below\n  useEffect(() => {\n    onCloseRef.current = onClose\n  }, [onClose])\n\n  // On open focus the modal\n  useEffect(() => {\n    dialogRef.current?.focus()\n  }, [])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n\n      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node)\n      ) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // We need to reverse the array as the last opened modal should be the first to be with normal size\n  // while the first opened modal should shrink\n  const realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\n\n  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      css={backdropCss}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\n    >\n      <StyledDialog\n        css={dialogCss}\n        onKeyUp={handleKeyUp}\n        onKeyDown={handleFocusTrap}\n        className={className}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n        data-placement={placement}\n        data-size={size}\n        open\n        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
18
18
|
const StyledDialog = /* @__PURE__ */ _styled("dialog", process.env.NODE_ENV === "production" ? {
|
|
19
19
|
shouldForwardProp: (prop) => !["position", "size", "openedModals", "top"].includes(prop),
|
|
20
20
|
target: "e1fims040"
|
|
@@ -45,7 +45,7 @@ const StyledDialog = /* @__PURE__ */ _styled("dialog", process.env.NODE_ENV ===
|
|
|
45
45
|
}) => position > 0 ? `
|
|
46
46
|
width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;
|
|
47
47
|
transform: translate3d(0, -${top}px, 0);
|
|
48
|
-
` : void 0, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx"],"names":[],"mappings":"AAuDqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  opacity: 0;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n\n  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n  &[data-animation='true'] {\n    overflow: hidden;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].includes(prop),\n})<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}rem;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\n}\n\nexport const Dialog = ({\n  children,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n  const {\n    registerModal,\n    unregisterModal,\n    openedModals,\n    previsousOpenedModales,\n  } = useModal()\n\n  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\n\n    return () => {\n      unregisterModal(id)\n    }\n  }, [id, registerModal, unregisterModal])\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    document.body.appendChild(element)\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [])\n\n  // Save the reassignment of eventHandler in the useEffect below\n  useEffect(() => {\n    onCloseRef.current = onClose\n  }, [onClose])\n\n  // On open focus the modal\n  useEffect(() => {\n    dialogRef.current?.focus()\n  }, [])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // Stop click to prevent unexpected dialog close\n  const stopClick: MouseEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (hideOnClickOutside) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // We need to reverse the array as the last opened modal should be the first to be with normal size\n  // while the first opened modal should shrink\n  const realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\n\n  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      css={backdropCss}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\n    >\n      <StyledDialog\n        css={dialogCss}\n        onKeyUp={handleKeyUp}\n        onKeyDown={handleFocusTrap}\n        className={className}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n        data-placement={placement}\n        data-size={size}\n        open\n        onClick={stopClick}\n        onCancel={stopCancel}\n        onClose={stopCancel}\n        onMouseDown={stopClick}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
48
|
+
` : void 0, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx"],"names":[],"mappings":"AAuDqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  opacity: 0;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n\n  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n  &[data-animation='true'] {\n    overflow: hidden;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].includes(prop),\n})<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}rem;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\n}\n\nexport const Dialog = ({\n  children,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n  const {\n    registerModal,\n    unregisterModal,\n    openedModals,\n    previsousOpenedModales,\n  } = useModal()\n\n  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\n\n    return () => {\n      unregisterModal(id)\n    }\n  }, [id, registerModal, unregisterModal])\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    document.body.appendChild(element)\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [])\n\n  // Save the reassignment of eventHandler in the useEffect below\n  useEffect(() => {\n    onCloseRef.current = onClose\n  }, [onClose])\n\n  // On open focus the modal\n  useEffect(() => {\n    dialogRef.current?.focus()\n  }, [])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n\n      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node)\n      ) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // We need to reverse the array as the last opened modal should be the first to be with normal size\n  // while the first opened modal should shrink\n  const realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\n\n  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      css={backdropCss}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\n    >\n      <StyledDialog\n        css={dialogCss}\n        onKeyUp={handleKeyUp}\n        onKeyDown={handleFocusTrap}\n        className={className}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n        data-placement={placement}\n        data-size={size}\n        open\n        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
49
49
|
const stopCancel = (event) => {
|
|
50
50
|
event.preventDefault();
|
|
51
51
|
event.stopPropagation();
|
|
@@ -115,9 +115,6 @@ const Dialog = ({
|
|
|
115
115
|
const stopFocus = useCallback((event) => {
|
|
116
116
|
event.stopPropagation();
|
|
117
117
|
}, []);
|
|
118
|
-
const stopClick = useCallback((event) => {
|
|
119
|
-
event.stopPropagation();
|
|
120
|
-
}, []);
|
|
121
118
|
const handleKeyUp = useCallback((event) => {
|
|
122
119
|
event.stopPropagation();
|
|
123
120
|
if (event.key === "Escape" && hideOnEsc) {
|
|
@@ -127,7 +124,7 @@ const Dialog = ({
|
|
|
127
124
|
}, [hideOnEsc]);
|
|
128
125
|
const handleClose = useCallback((event) => {
|
|
129
126
|
event.stopPropagation();
|
|
130
|
-
if (hideOnClickOutside) {
|
|
127
|
+
if (hideOnClickOutside && dialogRef.current && !dialogRef.current.contains(event.target)) {
|
|
131
128
|
onCloseRef.current();
|
|
132
129
|
} else {
|
|
133
130
|
dialogRef.current?.focus();
|
|
@@ -168,7 +165,7 @@ const Dialog = ({
|
|
|
168
165
|
top = (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 - currentModalHeight / 2 + 20;
|
|
169
166
|
}
|
|
170
167
|
const animation = openedModals.length > 1 && position === 0 && previsousOpenedModales.length < openedModals.length;
|
|
171
|
-
return createPortal(/* @__PURE__ */ jsx(StyledBackdrop, { "data-open": true, onClick: handleClose, className: backdropClassName, css: backdropCss, "data-testid": dataTestId ? `${dataTestId}-backdrop` : void 0, onFocus: stopFocus, "data-animation": animation, "data-visible": isVisible, id: "backdrop-modal", children: /* @__PURE__ */ jsx(StyledDialog, { css: dialogCss, onKeyUp: handleKeyUp, onKeyDown: handleFocusTrap, className, id, "data-testid": dataTestId, "aria-label": ariaLabel, "data-placement": placement, "data-size": size, open: true,
|
|
168
|
+
return createPortal(/* @__PURE__ */ jsx(StyledBackdrop, { "data-open": true, onClick: handleClose, className: backdropClassName, css: backdropCss, "data-testid": dataTestId ? `${dataTestId}-backdrop` : void 0, onFocus: stopFocus, "data-animation": animation, "data-visible": isVisible, id: "backdrop-modal", children: /* @__PURE__ */ jsx(StyledDialog, { css: dialogCss, onKeyUp: handleKeyUp, onKeyDown: handleFocusTrap, className, id, "data-testid": dataTestId, "aria-label": ariaLabel, "data-placement": placement, "data-size": size, open: true, onCancel: stopCancel, onClose: stopCancel, "aria-modal": true, ref: dialogRef, tabIndex: 0, position, top: Math.max(top, 0), "data-animation": animation, size, children }) }), containerRef.current);
|
|
172
169
|
};
|
|
173
170
|
export {
|
|
174
171
|
Dialog,
|