@ultraviolet/plus 0.15.2 → 0.15.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/CodeEditor/CodeEditor.cjs +39 -0
- package/dist/components/ContentCard/Skeleton.cjs +43 -0
- package/dist/components/ContentCard/index.cjs +154 -0
- package/dist/components/ContentCardGroup/Card.cjs +93 -0
- package/dist/components/ContentCardGroup/SkeletonCard.cjs +34 -0
- package/dist/components/ContentCardGroup/index.cjs +33 -0
- package/dist/components/Conversation/index.cjs +128 -0
- package/dist/components/CustomerSatisfaction/assets/1-5.svg.cjs +3 -0
- package/dist/components/CustomerSatisfaction/assets/1-5NB.svg.cjs +3 -0
- package/dist/components/CustomerSatisfaction/assets/2-5.svg.cjs +3 -0
- package/dist/components/CustomerSatisfaction/assets/2-5NB.svg.cjs +3 -0
- package/dist/components/CustomerSatisfaction/assets/3-5.svg.cjs +3 -0
- package/dist/components/CustomerSatisfaction/assets/3-5NB.svg.cjs +3 -0
- package/dist/components/CustomerSatisfaction/assets/4-5.svg.cjs +3 -0
- package/dist/components/CustomerSatisfaction/assets/4-5NB.svg.cjs +3 -0
- package/dist/components/CustomerSatisfaction/assets/5-5.svg.cjs +3 -0
- package/dist/components/CustomerSatisfaction/assets/5-5NB.svg.cjs +3 -0
- package/dist/components/CustomerSatisfaction/index.cjs +123 -0
- package/dist/components/EstimateCost/Components/CustomUnitInput.cjs +33 -0
- package/dist/components/EstimateCost/Components/Item.cjs +303 -0
- package/dist/components/EstimateCost/Components/LineThrough.cjs +18 -0
- package/dist/components/EstimateCost/Components/NumberInput.cjs +29 -0
- package/dist/components/EstimateCost/Components/Region.cjs +41 -0
- package/dist/components/EstimateCost/Components/Regular.cjs +35 -0
- package/dist/components/EstimateCost/Components/Strong.cjs +28 -0
- package/dist/components/EstimateCost/Components/Unit.cjs +50 -0
- package/dist/components/EstimateCost/Components/UnitInput.cjs +90 -0
- package/dist/components/EstimateCost/Components/Zone.cjs +40 -0
- package/dist/components/EstimateCost/EstimateCost.cjs +88 -0
- package/dist/components/EstimateCost/EstimateCostContent.cjs +245 -0
- package/dist/components/EstimateCost/EstimateCostProvider.cjs +36 -0
- package/dist/components/EstimateCost/OverlayComponent.cjs +122 -0
- package/dist/components/EstimateCost/OverlayContext.cjs +14 -0
- package/dist/components/EstimateCost/componentStyle.cjs +230 -0
- package/dist/components/EstimateCost/constants.cjs +30 -0
- package/dist/components/EstimateCost/helper.cjs +19 -0
- package/dist/components/EstimateCost/locales/en.cjs +23 -0
- package/dist/components/FAQ/index.cjs +36 -0
- package/dist/components/Navigation/Navigation.cjs +20 -0
- package/dist/components/Navigation/Navigation.d.ts +4 -3
- package/dist/components/Navigation/NavigationContent.cjs +216 -0
- package/dist/components/Navigation/NavigationProvider.cjs +111 -0
- package/dist/components/Navigation/NavigationProvider.d.ts +7 -18
- package/dist/components/Navigation/NavigationProvider.js +18 -16
- package/dist/components/Navigation/components/Group.cjs +57 -0
- package/dist/components/Navigation/components/Item.cjs +438 -0
- package/dist/components/Navigation/components/Item.d.ts +4 -3
- package/dist/components/Navigation/components/Item.js +34 -23
- package/dist/components/Navigation/components/PinnedItems.cjs +108 -0
- package/dist/components/Navigation/components/PinnedItems.d.ts +5 -1
- package/dist/components/Navigation/components/PinnedItems.js +11 -8
- package/dist/components/Navigation/components/Separator.cjs +17 -0
- package/dist/components/Navigation/constants.cjs +37 -0
- package/dist/components/Navigation/constants.d.ts +14 -0
- package/dist/components/Navigation/index.d.ts +1 -0
- package/dist/components/Navigation/locales/en.cjs +11 -0
- package/dist/components/SteppedListCard/Step.cjs +37 -0
- package/dist/components/SteppedListCard/SteppedListContainer.cjs +80 -0
- package/dist/components/SteppedListCard/SteppedListContent.cjs +67 -0
- package/dist/components/SteppedListCard/helper.cjs +37 -0
- package/dist/components/index.d.ts +1 -0
- package/dist/index.cjs +25 -0
- package/package.json +23 -7
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
const EstimateCostLocales = {
|
|
3
|
+
"estimate.cost.description": "This summary provides a cost estimation based on your configuration, the amount of time you expect to use the resource for, and the scale of your expected usage. For the purposes of this calculation, we consider that 1 month equals to 730 hours.",
|
|
4
|
+
"estimate.cost.label": "Estimated cost",
|
|
5
|
+
"estimate.cost.beta.free": "Free During Beta",
|
|
6
|
+
"estimate.cost.beta.discount": "% off during Beta",
|
|
7
|
+
"estimate.cost.beta.badge": "Beta Version",
|
|
8
|
+
"estimate.cost.units.seconds.label": "Second(s)",
|
|
9
|
+
"estimate.cost.units.minutes.label": "Minute(s)",
|
|
10
|
+
"estimate.cost.units.hours.label": "Hour(s)",
|
|
11
|
+
"estimate.cost.units.days.label": "Day(s)",
|
|
12
|
+
"estimate.cost.units.months.label": "Month(s)",
|
|
13
|
+
"estimate.cost.units.years.label": "Year(s)",
|
|
14
|
+
"estimate.cost.fees.oneTime.title": "One time fees",
|
|
15
|
+
"estimate.cost.fees.monthly.title": "Each month",
|
|
16
|
+
"estimate.cost.fees.commitment": "Commitment Fees",
|
|
17
|
+
"estimate.cost.units.gb.label": "GB",
|
|
18
|
+
"estimate.cost.notDefined": "Not defined",
|
|
19
|
+
"estimate.cost.submit.label": "Create",
|
|
20
|
+
"estimate.cost.region.label": "Region",
|
|
21
|
+
"estimate.cost.az.label": "Availability Zone"
|
|
22
|
+
};
|
|
23
|
+
module.exports = EstimateCostLocales;
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
3
|
+
const jsxRuntime = require("@emotion/react/jsx-runtime");
|
|
4
|
+
const _styled = require("@emotion/styled/base");
|
|
5
|
+
const icons = require("@ultraviolet/icons");
|
|
6
|
+
const ui = require("@ultraviolet/ui");
|
|
7
|
+
const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
|
|
8
|
+
const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
|
|
9
|
+
const StyledCard = /* @__PURE__ */ _styled__default.default(ui.Card, process.env.NODE_ENV === "production" ? {
|
|
10
|
+
target: "ebaosvo0"
|
|
11
|
+
} : {
|
|
12
|
+
target: "ebaosvo0",
|
|
13
|
+
label: "StyledCard"
|
|
14
|
+
})("padding:", ({
|
|
15
|
+
theme
|
|
16
|
+
}) => theme.space["2"], ";text-align:left;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3BsdXMvc3JjL2NvbXBvbmVudHMvRkFRL2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFLK0IiLCJmaWxlIjoiL2hvbWUvcnVubmVyL3dvcmsvdWx0cmF2aW9sZXQvdWx0cmF2aW9sZXQvcGFja2FnZXMvcGx1cy9zcmMvY29tcG9uZW50cy9GQVEvaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgeyBQcm9kdWN0SWNvbiB9IGZyb20gJ0B1bHRyYXZpb2xldC9pY29ucydcbmltcG9ydCB7IEJ1bGxldCwgQ2FyZCwgU3RhY2ssIFRleHQgfSBmcm9tICdAdWx0cmF2aW9sZXQvdWknXG5pbXBvcnQgdHlwZSB7IENvbXBvbmVudFByb3BzIH0gZnJvbSAncmVhY3QnXG5cbmNvbnN0IFN0eWxlZENhcmQgPSBzdHlsZWQoQ2FyZClgXG4gIHBhZGRpbmc6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuc3BhY2VbJzInXX07XG4gIHRleHQtYWxpZ246IGxlZnQ7XG5gXG5cbnR5cGUgRkFRUHJvcHMgPSB7XG4gIGRlc2NyaXB0aW9uOiBzdHJpbmdcbiAgcHJvZHVjdEljb25OYW1lPzogQ29tcG9uZW50UHJvcHM8dHlwZW9mIFByb2R1Y3RJY29uPlsnbmFtZSddXG4gIGlsbHVzdHJhdGlvblRleHQ/OiBudW1iZXIgfCBzdHJpbmdcbiAgbm90ZXM/OiBzdHJpbmdcbiAgdGl0bGU6IHN0cmluZ1xufVxuXG5leHBvcnQgY29uc3QgRkFRID0gKHtcbiAgcHJvZHVjdEljb25OYW1lLFxuICBpbGx1c3RyYXRpb25UZXh0LFxuICB0aXRsZSxcbiAgZGVzY3JpcHRpb24sXG4gIG5vdGVzLFxufTogRkFRUHJvcHMpID0+IChcbiAgPFN0eWxlZENhcmQ+XG4gICAgPFN0YWNrIGdhcD17Mn0gZGlyZWN0aW9uPVwicm93XCI+XG4gICAgICA8ZGl2PlxuICAgICAgICB7IXByb2R1Y3RJY29uTmFtZSAmJiBpbGx1c3RyYXRpb25UZXh0ID8gKFxuICAgICAgICAgIDxCdWxsZXQgc2VudGltZW50PVwicHJpbWFyeVwiIHRleHQ9e2lsbHVzdHJhdGlvblRleHQudG9TdHJpbmcoKX0gLz5cbiAgICAgICAgKSA6IG51bGx9XG4gICAgICAgIHtwcm9kdWN0SWNvbk5hbWUgPyAoXG4gICAgICAgICAgPFByb2R1Y3RJY29uIG5hbWU9e3Byb2R1Y3RJY29uTmFtZX0gc2l6ZT1cInhsYXJnZVwiIC8+XG4gICAgICAgICkgOiBudWxsfVxuICAgICAgPC9kaXY+XG4gICAgICA8ZGl2PlxuICAgICAgICA8VGV4dCBhcz1cInBcIiB2YXJpYW50PVwiYm9keVN0cm9uZ2VyXCIgcHJvbWluZW5jZT1cInN0cm9uZ1wiPlxuICAgICAgICAgIHt0aXRsZX1cbiAgICAgICAgPC9UZXh0PlxuICAgICAgICA8VGV4dCBhcz1cImRpdlwiIHZhcmlhbnQ9XCJib2R5U21hbGxcIj5cbiAgICAgICAgICB7ZGVzY3JpcHRpb259XG4gICAgICAgICAge25vdGVzID8gKFxuICAgICAgICAgICAgPFRleHQgdmFyaWFudD1cImNhcHRpb25cIiBhcz1cInNtYWxsXCIgaXRhbGljPlxuICAgICAgICAgICAgICB7bm90ZXN9XG4gICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgKSA6IG51bGx9XG4gICAgICAgIDwvVGV4dD5cbiAgICAgIDwvZGl2PlxuICAgIDwvU3RhY2s+XG4gIDwvU3R5bGVkQ2FyZD5cbilcbiJdfQ== */"));
|
|
17
|
+
const FAQ = ({
|
|
18
|
+
productIconName,
|
|
19
|
+
illustrationText,
|
|
20
|
+
title,
|
|
21
|
+
description,
|
|
22
|
+
notes
|
|
23
|
+
}) => /* @__PURE__ */ jsxRuntime.jsx(StyledCard, { children: /* @__PURE__ */ jsxRuntime.jsxs(ui.Stack, { gap: 2, direction: "row", children: [
|
|
24
|
+
/* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
|
|
25
|
+
!productIconName && illustrationText ? /* @__PURE__ */ jsxRuntime.jsx(ui.Bullet, { sentiment: "primary", text: illustrationText.toString() }) : null,
|
|
26
|
+
productIconName ? /* @__PURE__ */ jsxRuntime.jsx(icons.ProductIcon, { name: productIconName, size: "xlarge" }) : null
|
|
27
|
+
] }),
|
|
28
|
+
/* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
|
|
29
|
+
/* @__PURE__ */ jsxRuntime.jsx(ui.Text, { as: "p", variant: "bodyStronger", prominence: "strong", children: title }),
|
|
30
|
+
/* @__PURE__ */ jsxRuntime.jsxs(ui.Text, { as: "div", variant: "bodySmall", children: [
|
|
31
|
+
description,
|
|
32
|
+
notes ? /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { variant: "caption", as: "small", italic: true, children: notes }) : null
|
|
33
|
+
] })
|
|
34
|
+
] })
|
|
35
|
+
] }) });
|
|
36
|
+
exports.FAQ = FAQ;
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
3
|
+
const jsxRuntime = require("@emotion/react/jsx-runtime");
|
|
4
|
+
const NavigationContent = require("./NavigationContent.cjs");
|
|
5
|
+
const Group = require("./components/Group.cjs");
|
|
6
|
+
const Item = require("./components/Item.cjs");
|
|
7
|
+
const PinnedItems = require("./components/PinnedItems.cjs");
|
|
8
|
+
const Separator = require("./components/Separator.cjs");
|
|
9
|
+
const Navigation = ({
|
|
10
|
+
children,
|
|
11
|
+
logo,
|
|
12
|
+
onWidthResize,
|
|
13
|
+
className,
|
|
14
|
+
id
|
|
15
|
+
}) => /* @__PURE__ */ jsxRuntime.jsx(NavigationContent.NavigationContent, { logo, className, onWidthResize, id, children });
|
|
16
|
+
Navigation.Group = Group.Group;
|
|
17
|
+
Navigation.Item = Item.Item;
|
|
18
|
+
Navigation.PinnedItems = PinnedItems.PinnedItems;
|
|
19
|
+
Navigation.Separator = Separator.Separator;
|
|
20
|
+
exports.Navigation = Navigation;
|
|
@@ -31,7 +31,7 @@ export declare const Navigation: {
|
|
|
31
31
|
* some part of your logo
|
|
32
32
|
*/
|
|
33
33
|
}) => import("@emotion/react/jsx-runtime").JSX.Element | null;
|
|
34
|
-
Item: ({ children, categoryIcon, categoryIconVariant, label, subLabel, badgeText, badgeSentiment, href, onClick,
|
|
34
|
+
Item: ({ children, categoryIcon, categoryIconVariant, label, subLabel, badgeText, badgeSentiment, href, onClick, onClickPinUnpin, toggle, active, noPinButton, type, hasParents, as, disabled, noExpand, index, id, }: {
|
|
35
35
|
children?: ReactNode;
|
|
36
36
|
categoryIcon?: "security" | "console" | "database" | "pin" | "billing" | "storage" | "baremetal" | "webHosting" | "vpc" | "useCase" | "toolsServices" | "serverless" | "observability" | "network" | "managedServices" | "iot" | "documentation" | "dedicatedServer" | "datacenter" | "containers" | "compute" | "ai" | "labs" | "devTools" | "applicationIntegration" | undefined;
|
|
37
37
|
categoryIconVariant?: "neutral" | "primary" | undefined;
|
|
@@ -42,7 +42,7 @@ export declare const Navigation: {
|
|
|
42
42
|
badgeSentiment?: import("node_modules/@ultraviolet/ui/dist/theme").Color | undefined;
|
|
43
43
|
href?: string | undefined;
|
|
44
44
|
onClick?: ((toggle?: boolean | undefined) => void) | undefined;
|
|
45
|
-
|
|
45
|
+
onClickPinUnpin?: ((parameters: import("./constants").PinUnPinType) => void) | undefined;
|
|
46
46
|
toggle?: boolean | undefined;
|
|
47
47
|
active?: boolean | undefined;
|
|
48
48
|
noPinButton?: boolean | undefined;
|
|
@@ -53,8 +53,9 @@ export declare const Navigation: {
|
|
|
53
53
|
noExpand?: boolean | undefined;
|
|
54
54
|
disabled?: boolean | undefined;
|
|
55
55
|
}) => import("@emotion/react/jsx-runtime").JSX.Element | null;
|
|
56
|
-
PinnedItems: ({ toggle }: {
|
|
56
|
+
PinnedItems: ({ toggle, onReorder }: {
|
|
57
57
|
toggle?: boolean | undefined;
|
|
58
|
+
onReorder?: ((pinnedItems: string[]) => void) | undefined;
|
|
58
59
|
}) => import("@emotion/react/jsx-runtime").JSX.Element | null;
|
|
59
60
|
Separator: () => import("@emotion/react/jsx-runtime").JSX.Element;
|
|
60
61
|
};
|
|
@@ -0,0 +1,216 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
3
|
+
const jsxRuntime = require("@emotion/react/jsx-runtime");
|
|
4
|
+
const _styled = require("@emotion/styled/base");
|
|
5
|
+
const ui = require("@ultraviolet/ui");
|
|
6
|
+
const react = require("react");
|
|
7
|
+
const NavigationProvider = require("./NavigationProvider.cjs");
|
|
8
|
+
const constants = require("./constants.cjs");
|
|
9
|
+
const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
|
|
10
|
+
const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
|
|
11
|
+
function _EMOTION_STRINGIFIED_CSS_ERROR__() {
|
|
12
|
+
return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop).";
|
|
13
|
+
}
|
|
14
|
+
const StyledNav = /* @__PURE__ */ _styled__default.default("nav", process.env.NODE_ENV === "production" ? {
|
|
15
|
+
target: "esezfu57"
|
|
16
|
+
} : {
|
|
17
|
+
target: "esezfu57",
|
|
18
|
+
label: "StyledNav"
|
|
19
|
+
})("display:flex;flex-direction:row;position:relative;border-right:1px solid ", ({
|
|
20
|
+
theme
|
|
21
|
+
}) => theme.colors.neutral.borderWeak, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/NavigationContent.tsx"],"names":[],"mappings":"AAY4B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/NavigationContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Button, Stack, Tooltip } from '@ultraviolet/ui'\nimport type { ReactNode } from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { useNavigation } from './NavigationProvider'\nimport {\n  ANIMATION_DURATION,\n  NAVIGATION_COLLASPED_WIDTH,\n  NAVIGATION_MAX_WIDTH,\n  NAVIGATION_MIN_WIDTH,\n} from './constants'\n\nconst StyledNav = styled.nav`\n  display: flex;\n  flex-direction: row;\n  position: relative;\n  border-right: 1px solid ${({ theme }) => theme.colors.neutral.borderWeak};\n`\n\nconst Container = styled('div', {\n  shouldForwardProp: prop => !['width'].includes(prop),\n})<{\n  width: number\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: column;\n\n  width: ${({ width }) => width}px;\n\n  &[data-expanded='true'][data-animation='false'] {\n    max-width: ${NAVIGATION_MAX_WIDTH}px;\n    min-width: ${NAVIGATION_MIN_WIDTH}px;\n  }\n\n  &[data-expanded='false'] {\n    width: ${NAVIGATION_COLLASPED_WIDTH}px;\n  }\n\n  &[data-animation='expand'] {\n    transition: width ${ANIMATION_DURATION}ms ease-in-out;\n    width: ${({ width }) => width}px;\n  }\n\n  &[data-animation='collapse'] {\n    transition: width ${ANIMATION_DURATION}ms ease-in-out;\n\n    width: ${NAVIGATION_COLLASPED_WIDTH}px;\n  }\n`\n\nconst StickyFooter = styled.div`\n  display: flex;\n  width: 100%;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-top: 1px solid ${({ theme }) => theme.colors.neutral.borderWeak};\n  padding: ${({ theme }) => `${theme.space['1']} ${theme.space['2']}`};\n  transition: justify-content ${ANIMATION_DURATION}ms ease-in-out;\n  box-shadow: ${({ theme }) => theme.shadows.defaultShadow};\n  transition: box-shadow 230ms ease-in-out;\n  justify-content: flex-end;\n\n  &[data-has-overflow-style='false'] {\n    box-shadow: none;\n    border: none;\n  }\n`\n\nconst Header = styled.div`\n  background: ${({ theme }) => theme.colors.neutral.background};\n`\n\nconst LogoContainer = styled(Stack)`\n  margin: ${({ theme }) =>\n    `${theme.space['3']} ${theme.space['3']} ${theme.space['2']} ${theme.space['3']}`};\n  max-width: 220px;\n  height: 22px;\n`\n\nconst ContentContainer = styled.div`\n  overflow: hidden;\n  display: flex;\n  flex-direction: column;\n  flex-grow: 1;\n`\n\nconst Content = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\n  flex-grow: 1;\n\n  &[data-is-expanded='false'] {\n    padding: ${({ theme }) => theme.space['2']} 0;\n  }\n\n  &[data-is-expanded='true'],\n  &[data-animation='expand'] {\n    padding: ${({ theme }) => theme.space['2']};\n  }\n`\n\nconst Slider = styled.div`\n  background: transparent;\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  right: 0;\n  width: 6px;\n  cursor: col-resize;\n  border-right: 2px solid transparent;\n  display: flex;\n\n  &:hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n`\n\ntype NavigationContentProps = {\n  children: ReactNode\n  logo?: ReactNode | ((expanded: boolean) => ReactNode)\n  className?: string\n  onWidthResize?: (width: number) => void\n  id?: string\n}\n\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n}: NavigationContentProps) => {\n  const context = useNavigation()\n\n  if (!context) {\n    throw new Error(\n      'Navigation should be inside NavigationProvider to use it properly.',\n    )\n  }\n\n  const {\n    setWidth,\n    width,\n    expanded,\n    toggleExpand,\n    animation,\n    locales,\n    navigationRef,\n  } = context\n\n  const sliderRef = useRef<HTMLDivElement>(null)\n  const contentRef = useRef<HTMLDivElement>(null)\n\n  const isScrollAtBottom = useCallback(() => {\n    if (contentRef.current) {\n      if (\n        contentRef.current.scrollTop + contentRef.current.offsetHeight >=\n        contentRef.current.scrollHeight\n      ) {\n        return false\n      }\n    }\n\n    return true\n  }, [])\n\n  const [footerHasOverflowStyle, setFooterHasOverflowStyle] =\n    useState(isScrollAtBottom())\n\n  // This is for detecting if there is scroll on the content and set the shadow on the footer\n  useEffect(() => {\n    const scroll = () => {\n      const hasOverflow = isScrollAtBottom()\n\n      if (footerHasOverflowStyle !== hasOverflow) {\n        setFooterHasOverflowStyle(hasOverflow)\n      }\n    }\n\n    if (contentRef.current) {\n      contentRef.current.addEventListener('scroll', scroll)\n    }\n\n    return () => {\n      // eslint-disable-next-line react-hooks/exhaustive-deps\n      contentRef.current?.removeEventListener('scroll', scroll)\n    }\n  }, [footerHasOverflowStyle, isScrollAtBottom])\n\n  // This will set the shadow on the footer when the component is mounted\n  useEffect(\n    () => {\n      setFooterHasOverflowStyle(isScrollAtBottom())\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [contentRef.current],\n  )\n\n  // It will handle the resize of the navigation when the user drag the vertical bar\n  useEffect(() => {\n    let prevX: number\n    let navRect: DOMRect | undefined\n    let shouldCollapseOnMouseUp = false\n    let shouldExpandOnMouseUp = false\n\n    const mouseMove = (event: MouseEvent) => {\n      if (prevX !== undefined) {\n        const navWidth = navRect?.width ?? 0\n        const newWidth = navWidth + (event.clientX - prevX)\n\n        if (navigationRef.current && expanded) {\n          navigationRef.current.style.width = `${newWidth}px`\n        }\n\n        if (newWidth <= NAVIGATION_MIN_WIDTH) {\n          shouldCollapseOnMouseUp = true\n        } else {\n          shouldCollapseOnMouseUp = false\n        }\n\n        if (newWidth >= NAVIGATION_COLLASPED_WIDTH && !expanded) {\n          shouldExpandOnMouseUp = true\n        } else {\n          shouldExpandOnMouseUp = false\n        }\n      }\n    }\n\n    const mousedown = (event: MouseEvent) => {\n      document.body.style.pointerEvents = 'none'\n      document.body.style.userSelect = 'none'\n\n      prevX = event.clientX\n      navRect = navigationRef.current?.getBoundingClientRect()\n\n      const mouseup = () => {\n        if (shouldCollapseOnMouseUp || shouldExpandOnMouseUp) {\n          toggleExpand()\n        }\n\n        if (navigationRef.current) {\n          if (!shouldCollapseOnMouseUp && !shouldExpandOnMouseUp) {\n            onWidthResize?.(navigationRef.current.offsetWidth)\n            setWidth?.(navigationRef.current.offsetWidth)\n          }\n\n          if (!expanded) {\n            navigationRef.current.style.width = ''\n          }\n        }\n\n        document.removeEventListener('mousemove', mouseMove)\n        window.removeEventListener('mouseup', mouseup)\n\n        document.body.style.pointerEvents = ''\n        document.body.style.userSelect = ''\n      }\n\n      document.addEventListener('mousemove', mouseMove)\n      window.addEventListener('mouseup', mouseup)\n    }\n\n    sliderRef.current?.addEventListener('mousedown', mousedown)\n\n    return () => {\n      // eslint-disable-next-line react-hooks/exhaustive-deps\n      sliderRef.current?.removeEventListener('mousedown', mousedown)\n    }\n  }, [expanded, navigationRef, onWidthResize, setWidth, toggleExpand])\n\n  return (\n    <StyledNav className={className} id={id}>\n      <Container\n        ref={navigationRef}\n        data-animation={animation}\n        data-expanded={expanded}\n        width={width}\n      >\n        <Header>\n          <LogoContainer\n            justifyContent={!expanded ? 'center' : undefined}\n            alignItems=\"start\"\n          >\n            {typeof logo === 'function'\n              ? logo(animation ? false : expanded)\n              : logo}\n          </LogoContainer>\n        </Header>\n        <ContentContainer>\n          <Content\n            ref={contentRef}\n            gap={0.25}\n            data-is-expanded={expanded}\n            data-animation={animation}\n          >\n            {children}\n          </Content>\n          <StickyFooter data-has-overflow-style={footerHasOverflowStyle}>\n            <Tooltip\n              text={\n                expanded\n                  ? locales['navigation.collapse.button']\n                  : locales['navigation.expand.button']\n              }\n              placement=\"right\"\n            >\n              <Button\n                variant=\"ghost\"\n                sentiment=\"neutral\"\n                size=\"small\"\n                icon={expanded ? 'arrow-left-double' : 'arrow-right-double'}\n                onClick={() => toggleExpand()}\n              />\n            </Tooltip>\n          </StickyFooter>\n        </ContentContainer>\n      </Container>\n      <Slider ref={sliderRef} />\n    </StyledNav>\n  )\n}\n"]} */"));
|
|
22
|
+
const Container = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
|
|
23
|
+
shouldForwardProp: (prop) => !["width"].includes(prop),
|
|
24
|
+
target: "esezfu56"
|
|
25
|
+
} : {
|
|
26
|
+
shouldForwardProp: (prop) => !["width"].includes(prop),
|
|
27
|
+
target: "esezfu56",
|
|
28
|
+
label: "Container"
|
|
29
|
+
})("background:", ({
|
|
30
|
+
theme
|
|
31
|
+
}) => theme.colors.neutral.background, ";display:flex;flex-direction:column;width:", ({
|
|
32
|
+
width
|
|
33
|
+
}) => width, "px;&[data-expanded='true'][data-animation='false']{max-width:", constants.NAVIGATION_MAX_WIDTH, "px;min-width:", constants.NAVIGATION_MIN_WIDTH, "px;}&[data-expanded='false']{width:", constants.NAVIGATION_COLLASPED_WIDTH, "px;}&[data-animation='expand']{transition:width ", constants.ANIMATION_DURATION, "ms ease-in-out;width:", ({
|
|
34
|
+
width
|
|
35
|
+
}) => width, "px;}&[data-animation='collapse']{transition:width ", constants.ANIMATION_DURATION, "ms ease-in-out;width:", constants.NAVIGATION_COLLASPED_WIDTH, "px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/NavigationContent.tsx"],"names":[],"mappings":"AAuBE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/NavigationContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Button, Stack, Tooltip } from '@ultraviolet/ui'\nimport type { ReactNode } from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { useNavigation } from './NavigationProvider'\nimport {\n  ANIMATION_DURATION,\n  NAVIGATION_COLLASPED_WIDTH,\n  NAVIGATION_MAX_WIDTH,\n  NAVIGATION_MIN_WIDTH,\n} from './constants'\n\nconst StyledNav = styled.nav`\n  display: flex;\n  flex-direction: row;\n  position: relative;\n  border-right: 1px solid ${({ theme }) => theme.colors.neutral.borderWeak};\n`\n\nconst Container = styled('div', {\n  shouldForwardProp: prop => !['width'].includes(prop),\n})<{\n  width: number\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: column;\n\n  width: ${({ width }) => width}px;\n\n  &[data-expanded='true'][data-animation='false'] {\n    max-width: ${NAVIGATION_MAX_WIDTH}px;\n    min-width: ${NAVIGATION_MIN_WIDTH}px;\n  }\n\n  &[data-expanded='false'] {\n    width: ${NAVIGATION_COLLASPED_WIDTH}px;\n  }\n\n  &[data-animation='expand'] {\n    transition: width ${ANIMATION_DURATION}ms ease-in-out;\n    width: ${({ width }) => width}px;\n  }\n\n  &[data-animation='collapse'] {\n    transition: width ${ANIMATION_DURATION}ms ease-in-out;\n\n    width: ${NAVIGATION_COLLASPED_WIDTH}px;\n  }\n`\n\nconst StickyFooter = styled.div`\n  display: flex;\n  width: 100%;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-top: 1px solid ${({ theme }) => theme.colors.neutral.borderWeak};\n  padding: ${({ theme }) => `${theme.space['1']} ${theme.space['2']}`};\n  transition: justify-content ${ANIMATION_DURATION}ms ease-in-out;\n  box-shadow: ${({ theme }) => theme.shadows.defaultShadow};\n  transition: box-shadow 230ms ease-in-out;\n  justify-content: flex-end;\n\n  &[data-has-overflow-style='false'] {\n    box-shadow: none;\n    border: none;\n  }\n`\n\nconst Header = styled.div`\n  background: ${({ theme }) => theme.colors.neutral.background};\n`\n\nconst LogoContainer = styled(Stack)`\n  margin: ${({ theme }) =>\n    `${theme.space['3']} ${theme.space['3']} ${theme.space['2']} ${theme.space['3']}`};\n  max-width: 220px;\n  height: 22px;\n`\n\nconst ContentContainer = styled.div`\n  overflow: hidden;\n  display: flex;\n  flex-direction: column;\n  flex-grow: 1;\n`\n\nconst Content = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\n  flex-grow: 1;\n\n  &[data-is-expanded='false'] {\n    padding: ${({ theme }) => theme.space['2']} 0;\n  }\n\n  &[data-is-expanded='true'],\n  &[data-animation='expand'] {\n    padding: ${({ theme }) => theme.space['2']};\n  }\n`\n\nconst Slider = styled.div`\n  background: transparent;\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  right: 0;\n  width: 6px;\n  cursor: col-resize;\n  border-right: 2px solid transparent;\n  display: flex;\n\n  &:hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n`\n\ntype NavigationContentProps = {\n  children: ReactNode\n  logo?: ReactNode | ((expanded: boolean) => ReactNode)\n  className?: string\n  onWidthResize?: (width: number) => void\n  id?: string\n}\n\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n}: NavigationContentProps) => {\n  const context = useNavigation()\n\n  if (!context) {\n    throw new Error(\n      'Navigation should be inside NavigationProvider to use it properly.',\n    )\n  }\n\n  const {\n    setWidth,\n    width,\n    expanded,\n    toggleExpand,\n    animation,\n    locales,\n    navigationRef,\n  } = context\n\n  const sliderRef = useRef<HTMLDivElement>(null)\n  const contentRef = useRef<HTMLDivElement>(null)\n\n  const isScrollAtBottom = useCallback(() => {\n    if (contentRef.current) {\n      if (\n        contentRef.current.scrollTop + contentRef.current.offsetHeight >=\n        contentRef.current.scrollHeight\n      ) {\n        return false\n      }\n    }\n\n    return true\n  }, [])\n\n  const [footerHasOverflowStyle, setFooterHasOverflowStyle] =\n    useState(isScrollAtBottom())\n\n  // This is for detecting if there is scroll on the content and set the shadow on the footer\n  useEffect(() => {\n    const scroll = () => {\n      const hasOverflow = isScrollAtBottom()\n\n      if (footerHasOverflowStyle !== hasOverflow) {\n        setFooterHasOverflowStyle(hasOverflow)\n      }\n    }\n\n    if (contentRef.current) {\n      contentRef.current.addEventListener('scroll', scroll)\n    }\n\n    return () => {\n      // eslint-disable-next-line react-hooks/exhaustive-deps\n      contentRef.current?.removeEventListener('scroll', scroll)\n    }\n  }, [footerHasOverflowStyle, isScrollAtBottom])\n\n  // This will set the shadow on the footer when the component is mounted\n  useEffect(\n    () => {\n      setFooterHasOverflowStyle(isScrollAtBottom())\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [contentRef.current],\n  )\n\n  // It will handle the resize of the navigation when the user drag the vertical bar\n  useEffect(() => {\n    let prevX: number\n    let navRect: DOMRect | undefined\n    let shouldCollapseOnMouseUp = false\n    let shouldExpandOnMouseUp = false\n\n    const mouseMove = (event: MouseEvent) => {\n      if (prevX !== undefined) {\n        const navWidth = navRect?.width ?? 0\n        const newWidth = navWidth + (event.clientX - prevX)\n\n        if (navigationRef.current && expanded) {\n          navigationRef.current.style.width = `${newWidth}px`\n        }\n\n        if (newWidth <= NAVIGATION_MIN_WIDTH) {\n          shouldCollapseOnMouseUp = true\n        } else {\n          shouldCollapseOnMouseUp = false\n        }\n\n        if (newWidth >= NAVIGATION_COLLASPED_WIDTH && !expanded) {\n          shouldExpandOnMouseUp = true\n        } else {\n          shouldExpandOnMouseUp = false\n        }\n      }\n    }\n\n    const mousedown = (event: MouseEvent) => {\n      document.body.style.pointerEvents = 'none'\n      document.body.style.userSelect = 'none'\n\n      prevX = event.clientX\n      navRect = navigationRef.current?.getBoundingClientRect()\n\n      const mouseup = () => {\n        if (shouldCollapseOnMouseUp || shouldExpandOnMouseUp) {\n          toggleExpand()\n        }\n\n        if (navigationRef.current) {\n          if (!shouldCollapseOnMouseUp && !shouldExpandOnMouseUp) {\n            onWidthResize?.(navigationRef.current.offsetWidth)\n            setWidth?.(navigationRef.current.offsetWidth)\n          }\n\n          if (!expanded) {\n            navigationRef.current.style.width = ''\n          }\n        }\n\n        document.removeEventListener('mousemove', mouseMove)\n        window.removeEventListener('mouseup', mouseup)\n\n        document.body.style.pointerEvents = ''\n        document.body.style.userSelect = ''\n      }\n\n      document.addEventListener('mousemove', mouseMove)\n      window.addEventListener('mouseup', mouseup)\n    }\n\n    sliderRef.current?.addEventListener('mousedown', mousedown)\n\n    return () => {\n      // eslint-disable-next-line react-hooks/exhaustive-deps\n      sliderRef.current?.removeEventListener('mousedown', mousedown)\n    }\n  }, [expanded, navigationRef, onWidthResize, setWidth, toggleExpand])\n\n  return (\n    <StyledNav className={className} id={id}>\n      <Container\n        ref={navigationRef}\n        data-animation={animation}\n        data-expanded={expanded}\n        width={width}\n      >\n        <Header>\n          <LogoContainer\n            justifyContent={!expanded ? 'center' : undefined}\n            alignItems=\"start\"\n          >\n            {typeof logo === 'function'\n              ? logo(animation ? false : expanded)\n              : logo}\n          </LogoContainer>\n        </Header>\n        <ContentContainer>\n          <Content\n            ref={contentRef}\n            gap={0.25}\n            data-is-expanded={expanded}\n            data-animation={animation}\n          >\n            {children}\n          </Content>\n          <StickyFooter data-has-overflow-style={footerHasOverflowStyle}>\n            <Tooltip\n              text={\n                expanded\n                  ? locales['navigation.collapse.button']\n                  : locales['navigation.expand.button']\n              }\n              placement=\"right\"\n            >\n              <Button\n                variant=\"ghost\"\n                sentiment=\"neutral\"\n                size=\"small\"\n                icon={expanded ? 'arrow-left-double' : 'arrow-right-double'}\n                onClick={() => toggleExpand()}\n              />\n            </Tooltip>\n          </StickyFooter>\n        </ContentContainer>\n      </Container>\n      <Slider ref={sliderRef} />\n    </StyledNav>\n  )\n}\n"]} */"));
|
|
36
|
+
const StickyFooter = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
|
|
37
|
+
target: "esezfu55"
|
|
38
|
+
} : {
|
|
39
|
+
target: "esezfu55",
|
|
40
|
+
label: "StickyFooter"
|
|
41
|
+
})("display:flex;width:100%;background:", ({
|
|
42
|
+
theme
|
|
43
|
+
}) => theme.colors.neutral.background, ";border-top:1px solid ", ({
|
|
44
|
+
theme
|
|
45
|
+
}) => theme.colors.neutral.borderWeak, ";padding:", ({
|
|
46
|
+
theme
|
|
47
|
+
}) => `${theme.space["1"]} ${theme.space["2"]}`, ";transition:justify-content ", constants.ANIMATION_DURATION, "ms ease-in-out;box-shadow:", ({
|
|
48
|
+
theme
|
|
49
|
+
}) => theme.shadows.defaultShadow, ";transition:box-shadow 230ms ease-in-out;justify-content:flex-end;&[data-has-overflow-style='false']{box-shadow:none;border:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/NavigationContent.tsx"],"names":[],"mappings":"AAmD+B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/NavigationContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Button, Stack, Tooltip } from '@ultraviolet/ui'\nimport type { ReactNode } from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { useNavigation } from './NavigationProvider'\nimport {\n  ANIMATION_DURATION,\n  NAVIGATION_COLLASPED_WIDTH,\n  NAVIGATION_MAX_WIDTH,\n  NAVIGATION_MIN_WIDTH,\n} from './constants'\n\nconst StyledNav = styled.nav`\n  display: flex;\n  flex-direction: row;\n  position: relative;\n  border-right: 1px solid ${({ theme }) => theme.colors.neutral.borderWeak};\n`\n\nconst Container = styled('div', {\n  shouldForwardProp: prop => !['width'].includes(prop),\n})<{\n  width: number\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: column;\n\n  width: ${({ width }) => width}px;\n\n  &[data-expanded='true'][data-animation='false'] {\n    max-width: ${NAVIGATION_MAX_WIDTH}px;\n    min-width: ${NAVIGATION_MIN_WIDTH}px;\n  }\n\n  &[data-expanded='false'] {\n    width: ${NAVIGATION_COLLASPED_WIDTH}px;\n  }\n\n  &[data-animation='expand'] {\n    transition: width ${ANIMATION_DURATION}ms ease-in-out;\n    width: ${({ width }) => width}px;\n  }\n\n  &[data-animation='collapse'] {\n    transition: width ${ANIMATION_DURATION}ms ease-in-out;\n\n    width: ${NAVIGATION_COLLASPED_WIDTH}px;\n  }\n`\n\nconst StickyFooter = styled.div`\n  display: flex;\n  width: 100%;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-top: 1px solid ${({ theme }) => theme.colors.neutral.borderWeak};\n  padding: ${({ theme }) => `${theme.space['1']} ${theme.space['2']}`};\n  transition: justify-content ${ANIMATION_DURATION}ms ease-in-out;\n  box-shadow: ${({ theme }) => theme.shadows.defaultShadow};\n  transition: box-shadow 230ms ease-in-out;\n  justify-content: flex-end;\n\n  &[data-has-overflow-style='false'] {\n    box-shadow: none;\n    border: none;\n  }\n`\n\nconst Header = styled.div`\n  background: ${({ theme }) => theme.colors.neutral.background};\n`\n\nconst LogoContainer = styled(Stack)`\n  margin: ${({ theme }) =>\n    `${theme.space['3']} ${theme.space['3']} ${theme.space['2']} ${theme.space['3']}`};\n  max-width: 220px;\n  height: 22px;\n`\n\nconst ContentContainer = styled.div`\n  overflow: hidden;\n  display: flex;\n  flex-direction: column;\n  flex-grow: 1;\n`\n\nconst Content = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\n  flex-grow: 1;\n\n  &[data-is-expanded='false'] {\n    padding: ${({ theme }) => theme.space['2']} 0;\n  }\n\n  &[data-is-expanded='true'],\n  &[data-animation='expand'] {\n    padding: ${({ theme }) => theme.space['2']};\n  }\n`\n\nconst Slider = styled.div`\n  background: transparent;\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  right: 0;\n  width: 6px;\n  cursor: col-resize;\n  border-right: 2px solid transparent;\n  display: flex;\n\n  &:hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n`\n\ntype NavigationContentProps = {\n  children: ReactNode\n  logo?: ReactNode | ((expanded: boolean) => ReactNode)\n  className?: string\n  onWidthResize?: (width: number) => void\n  id?: string\n}\n\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n}: NavigationContentProps) => {\n  const context = useNavigation()\n\n  if (!context) {\n    throw new Error(\n      'Navigation should be inside NavigationProvider to use it properly.',\n    )\n  }\n\n  const {\n    setWidth,\n    width,\n    expanded,\n    toggleExpand,\n    animation,\n    locales,\n    navigationRef,\n  } = context\n\n  const sliderRef = useRef<HTMLDivElement>(null)\n  const contentRef = useRef<HTMLDivElement>(null)\n\n  const isScrollAtBottom = useCallback(() => {\n    if (contentRef.current) {\n      if (\n        contentRef.current.scrollTop + contentRef.current.offsetHeight >=\n        contentRef.current.scrollHeight\n      ) {\n        return false\n      }\n    }\n\n    return true\n  }, [])\n\n  const [footerHasOverflowStyle, setFooterHasOverflowStyle] =\n    useState(isScrollAtBottom())\n\n  // This is for detecting if there is scroll on the content and set the shadow on the footer\n  useEffect(() => {\n    const scroll = () => {\n      const hasOverflow = isScrollAtBottom()\n\n      if (footerHasOverflowStyle !== hasOverflow) {\n        setFooterHasOverflowStyle(hasOverflow)\n      }\n    }\n\n    if (contentRef.current) {\n      contentRef.current.addEventListener('scroll', scroll)\n    }\n\n    return () => {\n      // eslint-disable-next-line react-hooks/exhaustive-deps\n      contentRef.current?.removeEventListener('scroll', scroll)\n    }\n  }, [footerHasOverflowStyle, isScrollAtBottom])\n\n  // This will set the shadow on the footer when the component is mounted\n  useEffect(\n    () => {\n      setFooterHasOverflowStyle(isScrollAtBottom())\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [contentRef.current],\n  )\n\n  // It will handle the resize of the navigation when the user drag the vertical bar\n  useEffect(() => {\n    let prevX: number\n    let navRect: DOMRect | undefined\n    let shouldCollapseOnMouseUp = false\n    let shouldExpandOnMouseUp = false\n\n    const mouseMove = (event: MouseEvent) => {\n      if (prevX !== undefined) {\n        const navWidth = navRect?.width ?? 0\n        const newWidth = navWidth + (event.clientX - prevX)\n\n        if (navigationRef.current && expanded) {\n          navigationRef.current.style.width = `${newWidth}px`\n        }\n\n        if (newWidth <= NAVIGATION_MIN_WIDTH) {\n          shouldCollapseOnMouseUp = true\n        } else {\n          shouldCollapseOnMouseUp = false\n        }\n\n        if (newWidth >= NAVIGATION_COLLASPED_WIDTH && !expanded) {\n          shouldExpandOnMouseUp = true\n        } else {\n          shouldExpandOnMouseUp = false\n        }\n      }\n    }\n\n    const mousedown = (event: MouseEvent) => {\n      document.body.style.pointerEvents = 'none'\n      document.body.style.userSelect = 'none'\n\n      prevX = event.clientX\n      navRect = navigationRef.current?.getBoundingClientRect()\n\n      const mouseup = () => {\n        if (shouldCollapseOnMouseUp || shouldExpandOnMouseUp) {\n          toggleExpand()\n        }\n\n        if (navigationRef.current) {\n          if (!shouldCollapseOnMouseUp && !shouldExpandOnMouseUp) {\n            onWidthResize?.(navigationRef.current.offsetWidth)\n            setWidth?.(navigationRef.current.offsetWidth)\n          }\n\n          if (!expanded) {\n            navigationRef.current.style.width = ''\n          }\n        }\n\n        document.removeEventListener('mousemove', mouseMove)\n        window.removeEventListener('mouseup', mouseup)\n\n        document.body.style.pointerEvents = ''\n        document.body.style.userSelect = ''\n      }\n\n      document.addEventListener('mousemove', mouseMove)\n      window.addEventListener('mouseup', mouseup)\n    }\n\n    sliderRef.current?.addEventListener('mousedown', mousedown)\n\n    return () => {\n      // eslint-disable-next-line react-hooks/exhaustive-deps\n      sliderRef.current?.removeEventListener('mousedown', mousedown)\n    }\n  }, [expanded, navigationRef, onWidthResize, setWidth, toggleExpand])\n\n  return (\n    <StyledNav className={className} id={id}>\n      <Container\n        ref={navigationRef}\n        data-animation={animation}\n        data-expanded={expanded}\n        width={width}\n      >\n        <Header>\n          <LogoContainer\n            justifyContent={!expanded ? 'center' : undefined}\n            alignItems=\"start\"\n          >\n            {typeof logo === 'function'\n              ? logo(animation ? false : expanded)\n              : logo}\n          </LogoContainer>\n        </Header>\n        <ContentContainer>\n          <Content\n            ref={contentRef}\n            gap={0.25}\n            data-is-expanded={expanded}\n            data-animation={animation}\n          >\n            {children}\n          </Content>\n          <StickyFooter data-has-overflow-style={footerHasOverflowStyle}>\n            <Tooltip\n              text={\n                expanded\n                  ? locales['navigation.collapse.button']\n                  : locales['navigation.expand.button']\n              }\n              placement=\"right\"\n            >\n              <Button\n                variant=\"ghost\"\n                sentiment=\"neutral\"\n                size=\"small\"\n                icon={expanded ? 'arrow-left-double' : 'arrow-right-double'}\n                onClick={() => toggleExpand()}\n              />\n            </Tooltip>\n          </StickyFooter>\n        </ContentContainer>\n      </Container>\n      <Slider ref={sliderRef} />\n    </StyledNav>\n  )\n}\n"]} */"));
|
|
50
|
+
const Header = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
|
|
51
|
+
target: "esezfu54"
|
|
52
|
+
} : {
|
|
53
|
+
target: "esezfu54",
|
|
54
|
+
label: "Header"
|
|
55
|
+
})("background:", ({
|
|
56
|
+
theme
|
|
57
|
+
}) => theme.colors.neutral.background, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/NavigationContent.tsx"],"names":[],"mappings":"AAoEyB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/NavigationContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Button, Stack, Tooltip } from '@ultraviolet/ui'\nimport type { ReactNode } from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { useNavigation } from './NavigationProvider'\nimport {\n  ANIMATION_DURATION,\n  NAVIGATION_COLLASPED_WIDTH,\n  NAVIGATION_MAX_WIDTH,\n  NAVIGATION_MIN_WIDTH,\n} from './constants'\n\nconst StyledNav = styled.nav`\n  display: flex;\n  flex-direction: row;\n  position: relative;\n  border-right: 1px solid ${({ theme }) => theme.colors.neutral.borderWeak};\n`\n\nconst Container = styled('div', {\n  shouldForwardProp: prop => !['width'].includes(prop),\n})<{\n  width: number\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: column;\n\n  width: ${({ width }) => width}px;\n\n  &[data-expanded='true'][data-animation='false'] {\n    max-width: ${NAVIGATION_MAX_WIDTH}px;\n    min-width: ${NAVIGATION_MIN_WIDTH}px;\n  }\n\n  &[data-expanded='false'] {\n    width: ${NAVIGATION_COLLASPED_WIDTH}px;\n  }\n\n  &[data-animation='expand'] {\n    transition: width ${ANIMATION_DURATION}ms ease-in-out;\n    width: ${({ width }) => width}px;\n  }\n\n  &[data-animation='collapse'] {\n    transition: width ${ANIMATION_DURATION}ms ease-in-out;\n\n    width: ${NAVIGATION_COLLASPED_WIDTH}px;\n  }\n`\n\nconst StickyFooter = styled.div`\n  display: flex;\n  width: 100%;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-top: 1px solid ${({ theme }) => theme.colors.neutral.borderWeak};\n  padding: ${({ theme }) => `${theme.space['1']} ${theme.space['2']}`};\n  transition: justify-content ${ANIMATION_DURATION}ms ease-in-out;\n  box-shadow: ${({ theme }) => theme.shadows.defaultShadow};\n  transition: box-shadow 230ms ease-in-out;\n  justify-content: flex-end;\n\n  &[data-has-overflow-style='false'] {\n    box-shadow: none;\n    border: none;\n  }\n`\n\nconst Header = styled.div`\n  background: ${({ theme }) => theme.colors.neutral.background};\n`\n\nconst LogoContainer = styled(Stack)`\n  margin: ${({ theme }) =>\n    `${theme.space['3']} ${theme.space['3']} ${theme.space['2']} ${theme.space['3']}`};\n  max-width: 220px;\n  height: 22px;\n`\n\nconst ContentContainer = styled.div`\n  overflow: hidden;\n  display: flex;\n  flex-direction: column;\n  flex-grow: 1;\n`\n\nconst Content = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\n  flex-grow: 1;\n\n  &[data-is-expanded='false'] {\n    padding: ${({ theme }) => theme.space['2']} 0;\n  }\n\n  &[data-is-expanded='true'],\n  &[data-animation='expand'] {\n    padding: ${({ theme }) => theme.space['2']};\n  }\n`\n\nconst Slider = styled.div`\n  background: transparent;\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  right: 0;\n  width: 6px;\n  cursor: col-resize;\n  border-right: 2px solid transparent;\n  display: flex;\n\n  &:hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n`\n\ntype NavigationContentProps = {\n  children: ReactNode\n  logo?: ReactNode | ((expanded: boolean) => ReactNode)\n  className?: string\n  onWidthResize?: (width: number) => void\n  id?: string\n}\n\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n}: NavigationContentProps) => {\n  const context = useNavigation()\n\n  if (!context) {\n    throw new Error(\n      'Navigation should be inside NavigationProvider to use it properly.',\n    )\n  }\n\n  const {\n    setWidth,\n    width,\n    expanded,\n    toggleExpand,\n    animation,\n    locales,\n    navigationRef,\n  } = context\n\n  const sliderRef = useRef<HTMLDivElement>(null)\n  const contentRef = useRef<HTMLDivElement>(null)\n\n  const isScrollAtBottom = useCallback(() => {\n    if (contentRef.current) {\n      if (\n        contentRef.current.scrollTop + contentRef.current.offsetHeight >=\n        contentRef.current.scrollHeight\n      ) {\n        return false\n      }\n    }\n\n    return true\n  }, [])\n\n  const [footerHasOverflowStyle, setFooterHasOverflowStyle] =\n    useState(isScrollAtBottom())\n\n  // This is for detecting if there is scroll on the content and set the shadow on the footer\n  useEffect(() => {\n    const scroll = () => {\n      const hasOverflow = isScrollAtBottom()\n\n      if (footerHasOverflowStyle !== hasOverflow) {\n        setFooterHasOverflowStyle(hasOverflow)\n      }\n    }\n\n    if (contentRef.current) {\n      contentRef.current.addEventListener('scroll', scroll)\n    }\n\n    return () => {\n      // eslint-disable-next-line react-hooks/exhaustive-deps\n      contentRef.current?.removeEventListener('scroll', scroll)\n    }\n  }, [footerHasOverflowStyle, isScrollAtBottom])\n\n  // This will set the shadow on the footer when the component is mounted\n  useEffect(\n    () => {\n      setFooterHasOverflowStyle(isScrollAtBottom())\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [contentRef.current],\n  )\n\n  // It will handle the resize of the navigation when the user drag the vertical bar\n  useEffect(() => {\n    let prevX: number\n    let navRect: DOMRect | undefined\n    let shouldCollapseOnMouseUp = false\n    let shouldExpandOnMouseUp = false\n\n    const mouseMove = (event: MouseEvent) => {\n      if (prevX !== undefined) {\n        const navWidth = navRect?.width ?? 0\n        const newWidth = navWidth + (event.clientX - prevX)\n\n        if (navigationRef.current && expanded) {\n          navigationRef.current.style.width = `${newWidth}px`\n        }\n\n        if (newWidth <= NAVIGATION_MIN_WIDTH) {\n          shouldCollapseOnMouseUp = true\n        } else {\n          shouldCollapseOnMouseUp = false\n        }\n\n        if (newWidth >= NAVIGATION_COLLASPED_WIDTH && !expanded) {\n          shouldExpandOnMouseUp = true\n        } else {\n          shouldExpandOnMouseUp = false\n        }\n      }\n    }\n\n    const mousedown = (event: MouseEvent) => {\n      document.body.style.pointerEvents = 'none'\n      document.body.style.userSelect = 'none'\n\n      prevX = event.clientX\n      navRect = navigationRef.current?.getBoundingClientRect()\n\n      const mouseup = () => {\n        if (shouldCollapseOnMouseUp || shouldExpandOnMouseUp) {\n          toggleExpand()\n        }\n\n        if (navigationRef.current) {\n          if (!shouldCollapseOnMouseUp && !shouldExpandOnMouseUp) {\n            onWidthResize?.(navigationRef.current.offsetWidth)\n            setWidth?.(navigationRef.current.offsetWidth)\n          }\n\n          if (!expanded) {\n            navigationRef.current.style.width = ''\n          }\n        }\n\n        document.removeEventListener('mousemove', mouseMove)\n        window.removeEventListener('mouseup', mouseup)\n\n        document.body.style.pointerEvents = ''\n        document.body.style.userSelect = ''\n      }\n\n      document.addEventListener('mousemove', mouseMove)\n      window.addEventListener('mouseup', mouseup)\n    }\n\n    sliderRef.current?.addEventListener('mousedown', mousedown)\n\n    return () => {\n      // eslint-disable-next-line react-hooks/exhaustive-deps\n      sliderRef.current?.removeEventListener('mousedown', mousedown)\n    }\n  }, [expanded, navigationRef, onWidthResize, setWidth, toggleExpand])\n\n  return (\n    <StyledNav className={className} id={id}>\n      <Container\n        ref={navigationRef}\n        data-animation={animation}\n        data-expanded={expanded}\n        width={width}\n      >\n        <Header>\n          <LogoContainer\n            justifyContent={!expanded ? 'center' : undefined}\n            alignItems=\"start\"\n          >\n            {typeof logo === 'function'\n              ? logo(animation ? false : expanded)\n              : logo}\n          </LogoContainer>\n        </Header>\n        <ContentContainer>\n          <Content\n            ref={contentRef}\n            gap={0.25}\n            data-is-expanded={expanded}\n            data-animation={animation}\n          >\n            {children}\n          </Content>\n          <StickyFooter data-has-overflow-style={footerHasOverflowStyle}>\n            <Tooltip\n              text={\n                expanded\n                  ? locales['navigation.collapse.button']\n                  : locales['navigation.expand.button']\n              }\n              placement=\"right\"\n            >\n              <Button\n                variant=\"ghost\"\n                sentiment=\"neutral\"\n                size=\"small\"\n                icon={expanded ? 'arrow-left-double' : 'arrow-right-double'}\n                onClick={() => toggleExpand()}\n              />\n            </Tooltip>\n          </StickyFooter>\n        </ContentContainer>\n      </Container>\n      <Slider ref={sliderRef} />\n    </StyledNav>\n  )\n}\n"]} */"));
|
|
58
|
+
const LogoContainer = /* @__PURE__ */ _styled__default.default(ui.Stack, process.env.NODE_ENV === "production" ? {
|
|
59
|
+
target: "esezfu53"
|
|
60
|
+
} : {
|
|
61
|
+
target: "esezfu53",
|
|
62
|
+
label: "LogoContainer"
|
|
63
|
+
})("margin:", ({
|
|
64
|
+
theme
|
|
65
|
+
}) => `${theme.space["3"]} ${theme.space["3"]} ${theme.space["2"]} ${theme.space["3"]}`, ";max-width:220px;height:22px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/NavigationContent.tsx"],"names":[],"mappings":"AAwEmC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/NavigationContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Button, Stack, Tooltip } from '@ultraviolet/ui'\nimport type { ReactNode } from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { useNavigation } from './NavigationProvider'\nimport {\n  ANIMATION_DURATION,\n  NAVIGATION_COLLASPED_WIDTH,\n  NAVIGATION_MAX_WIDTH,\n  NAVIGATION_MIN_WIDTH,\n} from './constants'\n\nconst StyledNav = styled.nav`\n  display: flex;\n  flex-direction: row;\n  position: relative;\n  border-right: 1px solid ${({ theme }) => theme.colors.neutral.borderWeak};\n`\n\nconst Container = styled('div', {\n  shouldForwardProp: prop => !['width'].includes(prop),\n})<{\n  width: number\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: column;\n\n  width: ${({ width }) => width}px;\n\n  &[data-expanded='true'][data-animation='false'] {\n    max-width: ${NAVIGATION_MAX_WIDTH}px;\n    min-width: ${NAVIGATION_MIN_WIDTH}px;\n  }\n\n  &[data-expanded='false'] {\n    width: ${NAVIGATION_COLLASPED_WIDTH}px;\n  }\n\n  &[data-animation='expand'] {\n    transition: width ${ANIMATION_DURATION}ms ease-in-out;\n    width: ${({ width }) => width}px;\n  }\n\n  &[data-animation='collapse'] {\n    transition: width ${ANIMATION_DURATION}ms ease-in-out;\n\n    width: ${NAVIGATION_COLLASPED_WIDTH}px;\n  }\n`\n\nconst StickyFooter = styled.div`\n  display: flex;\n  width: 100%;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-top: 1px solid ${({ theme }) => theme.colors.neutral.borderWeak};\n  padding: ${({ theme }) => `${theme.space['1']} ${theme.space['2']}`};\n  transition: justify-content ${ANIMATION_DURATION}ms ease-in-out;\n  box-shadow: ${({ theme }) => theme.shadows.defaultShadow};\n  transition: box-shadow 230ms ease-in-out;\n  justify-content: flex-end;\n\n  &[data-has-overflow-style='false'] {\n    box-shadow: none;\n    border: none;\n  }\n`\n\nconst Header = styled.div`\n  background: ${({ theme }) => theme.colors.neutral.background};\n`\n\nconst LogoContainer = styled(Stack)`\n  margin: ${({ theme }) =>\n    `${theme.space['3']} ${theme.space['3']} ${theme.space['2']} ${theme.space['3']}`};\n  max-width: 220px;\n  height: 22px;\n`\n\nconst ContentContainer = styled.div`\n  overflow: hidden;\n  display: flex;\n  flex-direction: column;\n  flex-grow: 1;\n`\n\nconst Content = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\n  flex-grow: 1;\n\n  &[data-is-expanded='false'] {\n    padding: ${({ theme }) => theme.space['2']} 0;\n  }\n\n  &[data-is-expanded='true'],\n  &[data-animation='expand'] {\n    padding: ${({ theme }) => theme.space['2']};\n  }\n`\n\nconst Slider = styled.div`\n  background: transparent;\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  right: 0;\n  width: 6px;\n  cursor: col-resize;\n  border-right: 2px solid transparent;\n  display: flex;\n\n  &:hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n`\n\ntype NavigationContentProps = {\n  children: ReactNode\n  logo?: ReactNode | ((expanded: boolean) => ReactNode)\n  className?: string\n  onWidthResize?: (width: number) => void\n  id?: string\n}\n\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n}: NavigationContentProps) => {\n  const context = useNavigation()\n\n  if (!context) {\n    throw new Error(\n      'Navigation should be inside NavigationProvider to use it properly.',\n    )\n  }\n\n  const {\n    setWidth,\n    width,\n    expanded,\n    toggleExpand,\n    animation,\n    locales,\n    navigationRef,\n  } = context\n\n  const sliderRef = useRef<HTMLDivElement>(null)\n  const contentRef = useRef<HTMLDivElement>(null)\n\n  const isScrollAtBottom = useCallback(() => {\n    if (contentRef.current) {\n      if (\n        contentRef.current.scrollTop + contentRef.current.offsetHeight >=\n        contentRef.current.scrollHeight\n      ) {\n        return false\n      }\n    }\n\n    return true\n  }, [])\n\n  const [footerHasOverflowStyle, setFooterHasOverflowStyle] =\n    useState(isScrollAtBottom())\n\n  // This is for detecting if there is scroll on the content and set the shadow on the footer\n  useEffect(() => {\n    const scroll = () => {\n      const hasOverflow = isScrollAtBottom()\n\n      if (footerHasOverflowStyle !== hasOverflow) {\n        setFooterHasOverflowStyle(hasOverflow)\n      }\n    }\n\n    if (contentRef.current) {\n      contentRef.current.addEventListener('scroll', scroll)\n    }\n\n    return () => {\n      // eslint-disable-next-line react-hooks/exhaustive-deps\n      contentRef.current?.removeEventListener('scroll', scroll)\n    }\n  }, [footerHasOverflowStyle, isScrollAtBottom])\n\n  // This will set the shadow on the footer when the component is mounted\n  useEffect(\n    () => {\n      setFooterHasOverflowStyle(isScrollAtBottom())\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [contentRef.current],\n  )\n\n  // It will handle the resize of the navigation when the user drag the vertical bar\n  useEffect(() => {\n    let prevX: number\n    let navRect: DOMRect | undefined\n    let shouldCollapseOnMouseUp = false\n    let shouldExpandOnMouseUp = false\n\n    const mouseMove = (event: MouseEvent) => {\n      if (prevX !== undefined) {\n        const navWidth = navRect?.width ?? 0\n        const newWidth = navWidth + (event.clientX - prevX)\n\n        if (navigationRef.current && expanded) {\n          navigationRef.current.style.width = `${newWidth}px`\n        }\n\n        if (newWidth <= NAVIGATION_MIN_WIDTH) {\n          shouldCollapseOnMouseUp = true\n        } else {\n          shouldCollapseOnMouseUp = false\n        }\n\n        if (newWidth >= NAVIGATION_COLLASPED_WIDTH && !expanded) {\n          shouldExpandOnMouseUp = true\n        } else {\n          shouldExpandOnMouseUp = false\n        }\n      }\n    }\n\n    const mousedown = (event: MouseEvent) => {\n      document.body.style.pointerEvents = 'none'\n      document.body.style.userSelect = 'none'\n\n      prevX = event.clientX\n      navRect = navigationRef.current?.getBoundingClientRect()\n\n      const mouseup = () => {\n        if (shouldCollapseOnMouseUp || shouldExpandOnMouseUp) {\n          toggleExpand()\n        }\n\n        if (navigationRef.current) {\n          if (!shouldCollapseOnMouseUp && !shouldExpandOnMouseUp) {\n            onWidthResize?.(navigationRef.current.offsetWidth)\n            setWidth?.(navigationRef.current.offsetWidth)\n          }\n\n          if (!expanded) {\n            navigationRef.current.style.width = ''\n          }\n        }\n\n        document.removeEventListener('mousemove', mouseMove)\n        window.removeEventListener('mouseup', mouseup)\n\n        document.body.style.pointerEvents = ''\n        document.body.style.userSelect = ''\n      }\n\n      document.addEventListener('mousemove', mouseMove)\n      window.addEventListener('mouseup', mouseup)\n    }\n\n    sliderRef.current?.addEventListener('mousedown', mousedown)\n\n    return () => {\n      // eslint-disable-next-line react-hooks/exhaustive-deps\n      sliderRef.current?.removeEventListener('mousedown', mousedown)\n    }\n  }, [expanded, navigationRef, onWidthResize, setWidth, toggleExpand])\n\n  return (\n    <StyledNav className={className} id={id}>\n      <Container\n        ref={navigationRef}\n        data-animation={animation}\n        data-expanded={expanded}\n        width={width}\n      >\n        <Header>\n          <LogoContainer\n            justifyContent={!expanded ? 'center' : undefined}\n            alignItems=\"start\"\n          >\n            {typeof logo === 'function'\n              ? logo(animation ? false : expanded)\n              : logo}\n          </LogoContainer>\n        </Header>\n        <ContentContainer>\n          <Content\n            ref={contentRef}\n            gap={0.25}\n            data-is-expanded={expanded}\n            data-animation={animation}\n          >\n            {children}\n          </Content>\n          <StickyFooter data-has-overflow-style={footerHasOverflowStyle}>\n            <Tooltip\n              text={\n                expanded\n                  ? locales['navigation.collapse.button']\n                  : locales['navigation.expand.button']\n              }\n              placement=\"right\"\n            >\n              <Button\n                variant=\"ghost\"\n                sentiment=\"neutral\"\n                size=\"small\"\n                icon={expanded ? 'arrow-left-double' : 'arrow-right-double'}\n                onClick={() => toggleExpand()}\n              />\n            </Tooltip>\n          </StickyFooter>\n        </ContentContainer>\n      </Container>\n      <Slider ref={sliderRef} />\n    </StyledNav>\n  )\n}\n"]} */"));
|
|
66
|
+
const ContentContainer = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
|
|
67
|
+
target: "esezfu52"
|
|
68
|
+
} : {
|
|
69
|
+
target: "esezfu52",
|
|
70
|
+
label: "ContentContainer"
|
|
71
|
+
})(process.env.NODE_ENV === "production" ? {
|
|
72
|
+
name: "12is9id",
|
|
73
|
+
styles: "overflow:hidden;display:flex;flex-direction:column;flex-grow:1"
|
|
74
|
+
} : {
|
|
75
|
+
name: "12is9id",
|
|
76
|
+
styles: "overflow:hidden;display:flex;flex-direction:column;flex-grow:1",
|
|
77
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/NavigationContent.tsx"],"names":[],"mappings":"AA+EmC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/NavigationContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Button, Stack, Tooltip } from '@ultraviolet/ui'\nimport type { ReactNode } from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { useNavigation } from './NavigationProvider'\nimport {\n  ANIMATION_DURATION,\n  NAVIGATION_COLLASPED_WIDTH,\n  NAVIGATION_MAX_WIDTH,\n  NAVIGATION_MIN_WIDTH,\n} from './constants'\n\nconst StyledNav = styled.nav`\n  display: flex;\n  flex-direction: row;\n  position: relative;\n  border-right: 1px solid ${({ theme }) => theme.colors.neutral.borderWeak};\n`\n\nconst Container = styled('div', {\n  shouldForwardProp: prop => !['width'].includes(prop),\n})<{\n  width: number\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: column;\n\n  width: ${({ width }) => width}px;\n\n  &[data-expanded='true'][data-animation='false'] {\n    max-width: ${NAVIGATION_MAX_WIDTH}px;\n    min-width: ${NAVIGATION_MIN_WIDTH}px;\n  }\n\n  &[data-expanded='false'] {\n    width: ${NAVIGATION_COLLASPED_WIDTH}px;\n  }\n\n  &[data-animation='expand'] {\n    transition: width ${ANIMATION_DURATION}ms ease-in-out;\n    width: ${({ width }) => width}px;\n  }\n\n  &[data-animation='collapse'] {\n    transition: width ${ANIMATION_DURATION}ms ease-in-out;\n\n    width: ${NAVIGATION_COLLASPED_WIDTH}px;\n  }\n`\n\nconst StickyFooter = styled.div`\n  display: flex;\n  width: 100%;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-top: 1px solid ${({ theme }) => theme.colors.neutral.borderWeak};\n  padding: ${({ theme }) => `${theme.space['1']} ${theme.space['2']}`};\n  transition: justify-content ${ANIMATION_DURATION}ms ease-in-out;\n  box-shadow: ${({ theme }) => theme.shadows.defaultShadow};\n  transition: box-shadow 230ms ease-in-out;\n  justify-content: flex-end;\n\n  &[data-has-overflow-style='false'] {\n    box-shadow: none;\n    border: none;\n  }\n`\n\nconst Header = styled.div`\n  background: ${({ theme }) => theme.colors.neutral.background};\n`\n\nconst LogoContainer = styled(Stack)`\n  margin: ${({ theme }) =>\n    `${theme.space['3']} ${theme.space['3']} ${theme.space['2']} ${theme.space['3']}`};\n  max-width: 220px;\n  height: 22px;\n`\n\nconst ContentContainer = styled.div`\n  overflow: hidden;\n  display: flex;\n  flex-direction: column;\n  flex-grow: 1;\n`\n\nconst Content = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\n  flex-grow: 1;\n\n  &[data-is-expanded='false'] {\n    padding: ${({ theme }) => theme.space['2']} 0;\n  }\n\n  &[data-is-expanded='true'],\n  &[data-animation='expand'] {\n    padding: ${({ theme }) => theme.space['2']};\n  }\n`\n\nconst Slider = styled.div`\n  background: transparent;\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  right: 0;\n  width: 6px;\n  cursor: col-resize;\n  border-right: 2px solid transparent;\n  display: flex;\n\n  &:hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n`\n\ntype NavigationContentProps = {\n  children: ReactNode\n  logo?: ReactNode | ((expanded: boolean) => ReactNode)\n  className?: string\n  onWidthResize?: (width: number) => void\n  id?: string\n}\n\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n}: NavigationContentProps) => {\n  const context = useNavigation()\n\n  if (!context) {\n    throw new Error(\n      'Navigation should be inside NavigationProvider to use it properly.',\n    )\n  }\n\n  const {\n    setWidth,\n    width,\n    expanded,\n    toggleExpand,\n    animation,\n    locales,\n    navigationRef,\n  } = context\n\n  const sliderRef = useRef<HTMLDivElement>(null)\n  const contentRef = useRef<HTMLDivElement>(null)\n\n  const isScrollAtBottom = useCallback(() => {\n    if (contentRef.current) {\n      if (\n        contentRef.current.scrollTop + contentRef.current.offsetHeight >=\n        contentRef.current.scrollHeight\n      ) {\n        return false\n      }\n    }\n\n    return true\n  }, [])\n\n  const [footerHasOverflowStyle, setFooterHasOverflowStyle] =\n    useState(isScrollAtBottom())\n\n  // This is for detecting if there is scroll on the content and set the shadow on the footer\n  useEffect(() => {\n    const scroll = () => {\n      const hasOverflow = isScrollAtBottom()\n\n      if (footerHasOverflowStyle !== hasOverflow) {\n        setFooterHasOverflowStyle(hasOverflow)\n      }\n    }\n\n    if (contentRef.current) {\n      contentRef.current.addEventListener('scroll', scroll)\n    }\n\n    return () => {\n      // eslint-disable-next-line react-hooks/exhaustive-deps\n      contentRef.current?.removeEventListener('scroll', scroll)\n    }\n  }, [footerHasOverflowStyle, isScrollAtBottom])\n\n  // This will set the shadow on the footer when the component is mounted\n  useEffect(\n    () => {\n      setFooterHasOverflowStyle(isScrollAtBottom())\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [contentRef.current],\n  )\n\n  // It will handle the resize of the navigation when the user drag the vertical bar\n  useEffect(() => {\n    let prevX: number\n    let navRect: DOMRect | undefined\n    let shouldCollapseOnMouseUp = false\n    let shouldExpandOnMouseUp = false\n\n    const mouseMove = (event: MouseEvent) => {\n      if (prevX !== undefined) {\n        const navWidth = navRect?.width ?? 0\n        const newWidth = navWidth + (event.clientX - prevX)\n\n        if (navigationRef.current && expanded) {\n          navigationRef.current.style.width = `${newWidth}px`\n        }\n\n        if (newWidth <= NAVIGATION_MIN_WIDTH) {\n          shouldCollapseOnMouseUp = true\n        } else {\n          shouldCollapseOnMouseUp = false\n        }\n\n        if (newWidth >= NAVIGATION_COLLASPED_WIDTH && !expanded) {\n          shouldExpandOnMouseUp = true\n        } else {\n          shouldExpandOnMouseUp = false\n        }\n      }\n    }\n\n    const mousedown = (event: MouseEvent) => {\n      document.body.style.pointerEvents = 'none'\n      document.body.style.userSelect = 'none'\n\n      prevX = event.clientX\n      navRect = navigationRef.current?.getBoundingClientRect()\n\n      const mouseup = () => {\n        if (shouldCollapseOnMouseUp || shouldExpandOnMouseUp) {\n          toggleExpand()\n        }\n\n        if (navigationRef.current) {\n          if (!shouldCollapseOnMouseUp && !shouldExpandOnMouseUp) {\n            onWidthResize?.(navigationRef.current.offsetWidth)\n            setWidth?.(navigationRef.current.offsetWidth)\n          }\n\n          if (!expanded) {\n            navigationRef.current.style.width = ''\n          }\n        }\n\n        document.removeEventListener('mousemove', mouseMove)\n        window.removeEventListener('mouseup', mouseup)\n\n        document.body.style.pointerEvents = ''\n        document.body.style.userSelect = ''\n      }\n\n      document.addEventListener('mousemove', mouseMove)\n      window.addEventListener('mouseup', mouseup)\n    }\n\n    sliderRef.current?.addEventListener('mousedown', mousedown)\n\n    return () => {\n      // eslint-disable-next-line react-hooks/exhaustive-deps\n      sliderRef.current?.removeEventListener('mousedown', mousedown)\n    }\n  }, [expanded, navigationRef, onWidthResize, setWidth, toggleExpand])\n\n  return (\n    <StyledNav className={className} id={id}>\n      <Container\n        ref={navigationRef}\n        data-animation={animation}\n        data-expanded={expanded}\n        width={width}\n      >\n        <Header>\n          <LogoContainer\n            justifyContent={!expanded ? 'center' : undefined}\n            alignItems=\"start\"\n          >\n            {typeof logo === 'function'\n              ? logo(animation ? false : expanded)\n              : logo}\n          </LogoContainer>\n        </Header>\n        <ContentContainer>\n          <Content\n            ref={contentRef}\n            gap={0.25}\n            data-is-expanded={expanded}\n            data-animation={animation}\n          >\n            {children}\n          </Content>\n          <StickyFooter data-has-overflow-style={footerHasOverflowStyle}>\n            <Tooltip\n              text={\n                expanded\n                  ? locales['navigation.collapse.button']\n                  : locales['navigation.expand.button']\n              }\n              placement=\"right\"\n            >\n              <Button\n                variant=\"ghost\"\n                sentiment=\"neutral\"\n                size=\"small\"\n                icon={expanded ? 'arrow-left-double' : 'arrow-right-double'}\n                onClick={() => toggleExpand()}\n              />\n            </Tooltip>\n          </StickyFooter>\n        </ContentContainer>\n      </Container>\n      <Slider ref={sliderRef} />\n    </StyledNav>\n  )\n}\n"]} */",
|
|
78
|
+
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
79
|
+
});
|
|
80
|
+
const Content = /* @__PURE__ */ _styled__default.default(ui.Stack, process.env.NODE_ENV === "production" ? {
|
|
81
|
+
target: "esezfu51"
|
|
82
|
+
} : {
|
|
83
|
+
target: "esezfu51",
|
|
84
|
+
label: "Content"
|
|
85
|
+
})("overflow-y:auto;overflow-x:hidden;flex-grow:1;&[data-is-expanded='false']{padding:", ({
|
|
86
|
+
theme
|
|
87
|
+
}) => theme.space["2"], " 0;}&[data-is-expanded='true'],&[data-animation='expand']{padding:", ({
|
|
88
|
+
theme
|
|
89
|
+
}) => theme.space["2"], ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/NavigationContent.tsx"],"names":[],"mappings":"AAsF6B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/NavigationContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Button, Stack, Tooltip } from '@ultraviolet/ui'\nimport type { ReactNode } from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { useNavigation } from './NavigationProvider'\nimport {\n  ANIMATION_DURATION,\n  NAVIGATION_COLLASPED_WIDTH,\n  NAVIGATION_MAX_WIDTH,\n  NAVIGATION_MIN_WIDTH,\n} from './constants'\n\nconst StyledNav = styled.nav`\n  display: flex;\n  flex-direction: row;\n  position: relative;\n  border-right: 1px solid ${({ theme }) => theme.colors.neutral.borderWeak};\n`\n\nconst Container = styled('div', {\n  shouldForwardProp: prop => !['width'].includes(prop),\n})<{\n  width: number\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: column;\n\n  width: ${({ width }) => width}px;\n\n  &[data-expanded='true'][data-animation='false'] {\n    max-width: ${NAVIGATION_MAX_WIDTH}px;\n    min-width: ${NAVIGATION_MIN_WIDTH}px;\n  }\n\n  &[data-expanded='false'] {\n    width: ${NAVIGATION_COLLASPED_WIDTH}px;\n  }\n\n  &[data-animation='expand'] {\n    transition: width ${ANIMATION_DURATION}ms ease-in-out;\n    width: ${({ width }) => width}px;\n  }\n\n  &[data-animation='collapse'] {\n    transition: width ${ANIMATION_DURATION}ms ease-in-out;\n\n    width: ${NAVIGATION_COLLASPED_WIDTH}px;\n  }\n`\n\nconst StickyFooter = styled.div`\n  display: flex;\n  width: 100%;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-top: 1px solid ${({ theme }) => theme.colors.neutral.borderWeak};\n  padding: ${({ theme }) => `${theme.space['1']} ${theme.space['2']}`};\n  transition: justify-content ${ANIMATION_DURATION}ms ease-in-out;\n  box-shadow: ${({ theme }) => theme.shadows.defaultShadow};\n  transition: box-shadow 230ms ease-in-out;\n  justify-content: flex-end;\n\n  &[data-has-overflow-style='false'] {\n    box-shadow: none;\n    border: none;\n  }\n`\n\nconst Header = styled.div`\n  background: ${({ theme }) => theme.colors.neutral.background};\n`\n\nconst LogoContainer = styled(Stack)`\n  margin: ${({ theme }) =>\n    `${theme.space['3']} ${theme.space['3']} ${theme.space['2']} ${theme.space['3']}`};\n  max-width: 220px;\n  height: 22px;\n`\n\nconst ContentContainer = styled.div`\n  overflow: hidden;\n  display: flex;\n  flex-direction: column;\n  flex-grow: 1;\n`\n\nconst Content = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\n  flex-grow: 1;\n\n  &[data-is-expanded='false'] {\n    padding: ${({ theme }) => theme.space['2']} 0;\n  }\n\n  &[data-is-expanded='true'],\n  &[data-animation='expand'] {\n    padding: ${({ theme }) => theme.space['2']};\n  }\n`\n\nconst Slider = styled.div`\n  background: transparent;\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  right: 0;\n  width: 6px;\n  cursor: col-resize;\n  border-right: 2px solid transparent;\n  display: flex;\n\n  &:hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n`\n\ntype NavigationContentProps = {\n  children: ReactNode\n  logo?: ReactNode | ((expanded: boolean) => ReactNode)\n  className?: string\n  onWidthResize?: (width: number) => void\n  id?: string\n}\n\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n}: NavigationContentProps) => {\n  const context = useNavigation()\n\n  if (!context) {\n    throw new Error(\n      'Navigation should be inside NavigationProvider to use it properly.',\n    )\n  }\n\n  const {\n    setWidth,\n    width,\n    expanded,\n    toggleExpand,\n    animation,\n    locales,\n    navigationRef,\n  } = context\n\n  const sliderRef = useRef<HTMLDivElement>(null)\n  const contentRef = useRef<HTMLDivElement>(null)\n\n  const isScrollAtBottom = useCallback(() => {\n    if (contentRef.current) {\n      if (\n        contentRef.current.scrollTop + contentRef.current.offsetHeight >=\n        contentRef.current.scrollHeight\n      ) {\n        return false\n      }\n    }\n\n    return true\n  }, [])\n\n  const [footerHasOverflowStyle, setFooterHasOverflowStyle] =\n    useState(isScrollAtBottom())\n\n  // This is for detecting if there is scroll on the content and set the shadow on the footer\n  useEffect(() => {\n    const scroll = () => {\n      const hasOverflow = isScrollAtBottom()\n\n      if (footerHasOverflowStyle !== hasOverflow) {\n        setFooterHasOverflowStyle(hasOverflow)\n      }\n    }\n\n    if (contentRef.current) {\n      contentRef.current.addEventListener('scroll', scroll)\n    }\n\n    return () => {\n      // eslint-disable-next-line react-hooks/exhaustive-deps\n      contentRef.current?.removeEventListener('scroll', scroll)\n    }\n  }, [footerHasOverflowStyle, isScrollAtBottom])\n\n  // This will set the shadow on the footer when the component is mounted\n  useEffect(\n    () => {\n      setFooterHasOverflowStyle(isScrollAtBottom())\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [contentRef.current],\n  )\n\n  // It will handle the resize of the navigation when the user drag the vertical bar\n  useEffect(() => {\n    let prevX: number\n    let navRect: DOMRect | undefined\n    let shouldCollapseOnMouseUp = false\n    let shouldExpandOnMouseUp = false\n\n    const mouseMove = (event: MouseEvent) => {\n      if (prevX !== undefined) {\n        const navWidth = navRect?.width ?? 0\n        const newWidth = navWidth + (event.clientX - prevX)\n\n        if (navigationRef.current && expanded) {\n          navigationRef.current.style.width = `${newWidth}px`\n        }\n\n        if (newWidth <= NAVIGATION_MIN_WIDTH) {\n          shouldCollapseOnMouseUp = true\n        } else {\n          shouldCollapseOnMouseUp = false\n        }\n\n        if (newWidth >= NAVIGATION_COLLASPED_WIDTH && !expanded) {\n          shouldExpandOnMouseUp = true\n        } else {\n          shouldExpandOnMouseUp = false\n        }\n      }\n    }\n\n    const mousedown = (event: MouseEvent) => {\n      document.body.style.pointerEvents = 'none'\n      document.body.style.userSelect = 'none'\n\n      prevX = event.clientX\n      navRect = navigationRef.current?.getBoundingClientRect()\n\n      const mouseup = () => {\n        if (shouldCollapseOnMouseUp || shouldExpandOnMouseUp) {\n          toggleExpand()\n        }\n\n        if (navigationRef.current) {\n          if (!shouldCollapseOnMouseUp && !shouldExpandOnMouseUp) {\n            onWidthResize?.(navigationRef.current.offsetWidth)\n            setWidth?.(navigationRef.current.offsetWidth)\n          }\n\n          if (!expanded) {\n            navigationRef.current.style.width = ''\n          }\n        }\n\n        document.removeEventListener('mousemove', mouseMove)\n        window.removeEventListener('mouseup', mouseup)\n\n        document.body.style.pointerEvents = ''\n        document.body.style.userSelect = ''\n      }\n\n      document.addEventListener('mousemove', mouseMove)\n      window.addEventListener('mouseup', mouseup)\n    }\n\n    sliderRef.current?.addEventListener('mousedown', mousedown)\n\n    return () => {\n      // eslint-disable-next-line react-hooks/exhaustive-deps\n      sliderRef.current?.removeEventListener('mousedown', mousedown)\n    }\n  }, [expanded, navigationRef, onWidthResize, setWidth, toggleExpand])\n\n  return (\n    <StyledNav className={className} id={id}>\n      <Container\n        ref={navigationRef}\n        data-animation={animation}\n        data-expanded={expanded}\n        width={width}\n      >\n        <Header>\n          <LogoContainer\n            justifyContent={!expanded ? 'center' : undefined}\n            alignItems=\"start\"\n          >\n            {typeof logo === 'function'\n              ? logo(animation ? false : expanded)\n              : logo}\n          </LogoContainer>\n        </Header>\n        <ContentContainer>\n          <Content\n            ref={contentRef}\n            gap={0.25}\n            data-is-expanded={expanded}\n            data-animation={animation}\n          >\n            {children}\n          </Content>\n          <StickyFooter data-has-overflow-style={footerHasOverflowStyle}>\n            <Tooltip\n              text={\n                expanded\n                  ? locales['navigation.collapse.button']\n                  : locales['navigation.expand.button']\n              }\n              placement=\"right\"\n            >\n              <Button\n                variant=\"ghost\"\n                sentiment=\"neutral\"\n                size=\"small\"\n                icon={expanded ? 'arrow-left-double' : 'arrow-right-double'}\n                onClick={() => toggleExpand()}\n              />\n            </Tooltip>\n          </StickyFooter>\n        </ContentContainer>\n      </Container>\n      <Slider ref={sliderRef} />\n    </StyledNav>\n  )\n}\n"]} */"));
|
|
90
|
+
const Slider = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
|
|
91
|
+
target: "esezfu50"
|
|
92
|
+
} : {
|
|
93
|
+
target: "esezfu50",
|
|
94
|
+
label: "Slider"
|
|
95
|
+
})("background:transparent;position:absolute;top:0;bottom:0;right:0;width:6px;cursor:col-resize;border-right:2px solid transparent;display:flex;&:hover{border-color:", ({
|
|
96
|
+
theme
|
|
97
|
+
}) => theme.colors.primary.border, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/NavigationContent.tsx"],"names":[],"mappings":"AAqGyB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/NavigationContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Button, Stack, Tooltip } from '@ultraviolet/ui'\nimport type { ReactNode } from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { useNavigation } from './NavigationProvider'\nimport {\n  ANIMATION_DURATION,\n  NAVIGATION_COLLASPED_WIDTH,\n  NAVIGATION_MAX_WIDTH,\n  NAVIGATION_MIN_WIDTH,\n} from './constants'\n\nconst StyledNav = styled.nav`\n  display: flex;\n  flex-direction: row;\n  position: relative;\n  border-right: 1px solid ${({ theme }) => theme.colors.neutral.borderWeak};\n`\n\nconst Container = styled('div', {\n  shouldForwardProp: prop => !['width'].includes(prop),\n})<{\n  width: number\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: column;\n\n  width: ${({ width }) => width}px;\n\n  &[data-expanded='true'][data-animation='false'] {\n    max-width: ${NAVIGATION_MAX_WIDTH}px;\n    min-width: ${NAVIGATION_MIN_WIDTH}px;\n  }\n\n  &[data-expanded='false'] {\n    width: ${NAVIGATION_COLLASPED_WIDTH}px;\n  }\n\n  &[data-animation='expand'] {\n    transition: width ${ANIMATION_DURATION}ms ease-in-out;\n    width: ${({ width }) => width}px;\n  }\n\n  &[data-animation='collapse'] {\n    transition: width ${ANIMATION_DURATION}ms ease-in-out;\n\n    width: ${NAVIGATION_COLLASPED_WIDTH}px;\n  }\n`\n\nconst StickyFooter = styled.div`\n  display: flex;\n  width: 100%;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-top: 1px solid ${({ theme }) => theme.colors.neutral.borderWeak};\n  padding: ${({ theme }) => `${theme.space['1']} ${theme.space['2']}`};\n  transition: justify-content ${ANIMATION_DURATION}ms ease-in-out;\n  box-shadow: ${({ theme }) => theme.shadows.defaultShadow};\n  transition: box-shadow 230ms ease-in-out;\n  justify-content: flex-end;\n\n  &[data-has-overflow-style='false'] {\n    box-shadow: none;\n    border: none;\n  }\n`\n\nconst Header = styled.div`\n  background: ${({ theme }) => theme.colors.neutral.background};\n`\n\nconst LogoContainer = styled(Stack)`\n  margin: ${({ theme }) =>\n    `${theme.space['3']} ${theme.space['3']} ${theme.space['2']} ${theme.space['3']}`};\n  max-width: 220px;\n  height: 22px;\n`\n\nconst ContentContainer = styled.div`\n  overflow: hidden;\n  display: flex;\n  flex-direction: column;\n  flex-grow: 1;\n`\n\nconst Content = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\n  flex-grow: 1;\n\n  &[data-is-expanded='false'] {\n    padding: ${({ theme }) => theme.space['2']} 0;\n  }\n\n  &[data-is-expanded='true'],\n  &[data-animation='expand'] {\n    padding: ${({ theme }) => theme.space['2']};\n  }\n`\n\nconst Slider = styled.div`\n  background: transparent;\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  right: 0;\n  width: 6px;\n  cursor: col-resize;\n  border-right: 2px solid transparent;\n  display: flex;\n\n  &:hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n`\n\ntype NavigationContentProps = {\n  children: ReactNode\n  logo?: ReactNode | ((expanded: boolean) => ReactNode)\n  className?: string\n  onWidthResize?: (width: number) => void\n  id?: string\n}\n\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n}: NavigationContentProps) => {\n  const context = useNavigation()\n\n  if (!context) {\n    throw new Error(\n      'Navigation should be inside NavigationProvider to use it properly.',\n    )\n  }\n\n  const {\n    setWidth,\n    width,\n    expanded,\n    toggleExpand,\n    animation,\n    locales,\n    navigationRef,\n  } = context\n\n  const sliderRef = useRef<HTMLDivElement>(null)\n  const contentRef = useRef<HTMLDivElement>(null)\n\n  const isScrollAtBottom = useCallback(() => {\n    if (contentRef.current) {\n      if (\n        contentRef.current.scrollTop + contentRef.current.offsetHeight >=\n        contentRef.current.scrollHeight\n      ) {\n        return false\n      }\n    }\n\n    return true\n  }, [])\n\n  const [footerHasOverflowStyle, setFooterHasOverflowStyle] =\n    useState(isScrollAtBottom())\n\n  // This is for detecting if there is scroll on the content and set the shadow on the footer\n  useEffect(() => {\n    const scroll = () => {\n      const hasOverflow = isScrollAtBottom()\n\n      if (footerHasOverflowStyle !== hasOverflow) {\n        setFooterHasOverflowStyle(hasOverflow)\n      }\n    }\n\n    if (contentRef.current) {\n      contentRef.current.addEventListener('scroll', scroll)\n    }\n\n    return () => {\n      // eslint-disable-next-line react-hooks/exhaustive-deps\n      contentRef.current?.removeEventListener('scroll', scroll)\n    }\n  }, [footerHasOverflowStyle, isScrollAtBottom])\n\n  // This will set the shadow on the footer when the component is mounted\n  useEffect(\n    () => {\n      setFooterHasOverflowStyle(isScrollAtBottom())\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [contentRef.current],\n  )\n\n  // It will handle the resize of the navigation when the user drag the vertical bar\n  useEffect(() => {\n    let prevX: number\n    let navRect: DOMRect | undefined\n    let shouldCollapseOnMouseUp = false\n    let shouldExpandOnMouseUp = false\n\n    const mouseMove = (event: MouseEvent) => {\n      if (prevX !== undefined) {\n        const navWidth = navRect?.width ?? 0\n        const newWidth = navWidth + (event.clientX - prevX)\n\n        if (navigationRef.current && expanded) {\n          navigationRef.current.style.width = `${newWidth}px`\n        }\n\n        if (newWidth <= NAVIGATION_MIN_WIDTH) {\n          shouldCollapseOnMouseUp = true\n        } else {\n          shouldCollapseOnMouseUp = false\n        }\n\n        if (newWidth >= NAVIGATION_COLLASPED_WIDTH && !expanded) {\n          shouldExpandOnMouseUp = true\n        } else {\n          shouldExpandOnMouseUp = false\n        }\n      }\n    }\n\n    const mousedown = (event: MouseEvent) => {\n      document.body.style.pointerEvents = 'none'\n      document.body.style.userSelect = 'none'\n\n      prevX = event.clientX\n      navRect = navigationRef.current?.getBoundingClientRect()\n\n      const mouseup = () => {\n        if (shouldCollapseOnMouseUp || shouldExpandOnMouseUp) {\n          toggleExpand()\n        }\n\n        if (navigationRef.current) {\n          if (!shouldCollapseOnMouseUp && !shouldExpandOnMouseUp) {\n            onWidthResize?.(navigationRef.current.offsetWidth)\n            setWidth?.(navigationRef.current.offsetWidth)\n          }\n\n          if (!expanded) {\n            navigationRef.current.style.width = ''\n          }\n        }\n\n        document.removeEventListener('mousemove', mouseMove)\n        window.removeEventListener('mouseup', mouseup)\n\n        document.body.style.pointerEvents = ''\n        document.body.style.userSelect = ''\n      }\n\n      document.addEventListener('mousemove', mouseMove)\n      window.addEventListener('mouseup', mouseup)\n    }\n\n    sliderRef.current?.addEventListener('mousedown', mousedown)\n\n    return () => {\n      // eslint-disable-next-line react-hooks/exhaustive-deps\n      sliderRef.current?.removeEventListener('mousedown', mousedown)\n    }\n  }, [expanded, navigationRef, onWidthResize, setWidth, toggleExpand])\n\n  return (\n    <StyledNav className={className} id={id}>\n      <Container\n        ref={navigationRef}\n        data-animation={animation}\n        data-expanded={expanded}\n        width={width}\n      >\n        <Header>\n          <LogoContainer\n            justifyContent={!expanded ? 'center' : undefined}\n            alignItems=\"start\"\n          >\n            {typeof logo === 'function'\n              ? logo(animation ? false : expanded)\n              : logo}\n          </LogoContainer>\n        </Header>\n        <ContentContainer>\n          <Content\n            ref={contentRef}\n            gap={0.25}\n            data-is-expanded={expanded}\n            data-animation={animation}\n          >\n            {children}\n          </Content>\n          <StickyFooter data-has-overflow-style={footerHasOverflowStyle}>\n            <Tooltip\n              text={\n                expanded\n                  ? locales['navigation.collapse.button']\n                  : locales['navigation.expand.button']\n              }\n              placement=\"right\"\n            >\n              <Button\n                variant=\"ghost\"\n                sentiment=\"neutral\"\n                size=\"small\"\n                icon={expanded ? 'arrow-left-double' : 'arrow-right-double'}\n                onClick={() => toggleExpand()}\n              />\n            </Tooltip>\n          </StickyFooter>\n        </ContentContainer>\n      </Container>\n      <Slider ref={sliderRef} />\n    </StyledNav>\n  )\n}\n"]} */"));
|
|
98
|
+
const NavigationContent = ({
|
|
99
|
+
children,
|
|
100
|
+
logo,
|
|
101
|
+
onWidthResize,
|
|
102
|
+
className,
|
|
103
|
+
id
|
|
104
|
+
}) => {
|
|
105
|
+
const context = NavigationProvider.useNavigation();
|
|
106
|
+
if (!context) {
|
|
107
|
+
throw new Error("Navigation should be inside NavigationProvider to use it properly.");
|
|
108
|
+
}
|
|
109
|
+
const {
|
|
110
|
+
setWidth,
|
|
111
|
+
width,
|
|
112
|
+
expanded,
|
|
113
|
+
toggleExpand,
|
|
114
|
+
animation,
|
|
115
|
+
locales,
|
|
116
|
+
navigationRef
|
|
117
|
+
} = context;
|
|
118
|
+
const sliderRef = react.useRef(null);
|
|
119
|
+
const contentRef = react.useRef(null);
|
|
120
|
+
const isScrollAtBottom = react.useCallback(() => {
|
|
121
|
+
if (contentRef.current) {
|
|
122
|
+
if (contentRef.current.scrollTop + contentRef.current.offsetHeight >= contentRef.current.scrollHeight) {
|
|
123
|
+
return false;
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
return true;
|
|
127
|
+
}, []);
|
|
128
|
+
const [footerHasOverflowStyle, setFooterHasOverflowStyle] = react.useState(isScrollAtBottom());
|
|
129
|
+
react.useEffect(() => {
|
|
130
|
+
const scroll = () => {
|
|
131
|
+
const hasOverflow = isScrollAtBottom();
|
|
132
|
+
if (footerHasOverflowStyle !== hasOverflow) {
|
|
133
|
+
setFooterHasOverflowStyle(hasOverflow);
|
|
134
|
+
}
|
|
135
|
+
};
|
|
136
|
+
if (contentRef.current) {
|
|
137
|
+
contentRef.current.addEventListener("scroll", scroll);
|
|
138
|
+
}
|
|
139
|
+
return () => {
|
|
140
|
+
contentRef.current?.removeEventListener("scroll", scroll);
|
|
141
|
+
};
|
|
142
|
+
}, [footerHasOverflowStyle, isScrollAtBottom]);
|
|
143
|
+
react.useEffect(
|
|
144
|
+
() => {
|
|
145
|
+
setFooterHasOverflowStyle(isScrollAtBottom());
|
|
146
|
+
},
|
|
147
|
+
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
148
|
+
[contentRef.current]
|
|
149
|
+
);
|
|
150
|
+
react.useEffect(() => {
|
|
151
|
+
let prevX;
|
|
152
|
+
let navRect;
|
|
153
|
+
let shouldCollapseOnMouseUp = false;
|
|
154
|
+
let shouldExpandOnMouseUp = false;
|
|
155
|
+
const mouseMove = (event) => {
|
|
156
|
+
if (prevX !== void 0) {
|
|
157
|
+
const navWidth = navRect?.width ?? 0;
|
|
158
|
+
const newWidth = navWidth + (event.clientX - prevX);
|
|
159
|
+
if (navigationRef.current && expanded) {
|
|
160
|
+
navigationRef.current.style.width = `${newWidth}px`;
|
|
161
|
+
}
|
|
162
|
+
if (newWidth <= constants.NAVIGATION_MIN_WIDTH) {
|
|
163
|
+
shouldCollapseOnMouseUp = true;
|
|
164
|
+
} else {
|
|
165
|
+
shouldCollapseOnMouseUp = false;
|
|
166
|
+
}
|
|
167
|
+
if (newWidth >= constants.NAVIGATION_COLLASPED_WIDTH && !expanded) {
|
|
168
|
+
shouldExpandOnMouseUp = true;
|
|
169
|
+
} else {
|
|
170
|
+
shouldExpandOnMouseUp = false;
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
};
|
|
174
|
+
const mousedown = (event) => {
|
|
175
|
+
document.body.style.pointerEvents = "none";
|
|
176
|
+
document.body.style.userSelect = "none";
|
|
177
|
+
prevX = event.clientX;
|
|
178
|
+
navRect = navigationRef.current?.getBoundingClientRect();
|
|
179
|
+
const mouseup = () => {
|
|
180
|
+
if (shouldCollapseOnMouseUp || shouldExpandOnMouseUp) {
|
|
181
|
+
toggleExpand();
|
|
182
|
+
}
|
|
183
|
+
if (navigationRef.current) {
|
|
184
|
+
if (!shouldCollapseOnMouseUp && !shouldExpandOnMouseUp) {
|
|
185
|
+
onWidthResize?.(navigationRef.current.offsetWidth);
|
|
186
|
+
setWidth?.(navigationRef.current.offsetWidth);
|
|
187
|
+
}
|
|
188
|
+
if (!expanded) {
|
|
189
|
+
navigationRef.current.style.width = "";
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
document.removeEventListener("mousemove", mouseMove);
|
|
193
|
+
window.removeEventListener("mouseup", mouseup);
|
|
194
|
+
document.body.style.pointerEvents = "";
|
|
195
|
+
document.body.style.userSelect = "";
|
|
196
|
+
};
|
|
197
|
+
document.addEventListener("mousemove", mouseMove);
|
|
198
|
+
window.addEventListener("mouseup", mouseup);
|
|
199
|
+
};
|
|
200
|
+
sliderRef.current?.addEventListener("mousedown", mousedown);
|
|
201
|
+
return () => {
|
|
202
|
+
sliderRef.current?.removeEventListener("mousedown", mousedown);
|
|
203
|
+
};
|
|
204
|
+
}, [expanded, navigationRef, onWidthResize, setWidth, toggleExpand]);
|
|
205
|
+
return /* @__PURE__ */ jsxRuntime.jsxs(StyledNav, { className, id, children: [
|
|
206
|
+
/* @__PURE__ */ jsxRuntime.jsxs(Container, { ref: navigationRef, "data-animation": animation, "data-expanded": expanded, width, children: [
|
|
207
|
+
/* @__PURE__ */ jsxRuntime.jsx(Header, { children: /* @__PURE__ */ jsxRuntime.jsx(LogoContainer, { justifyContent: !expanded ? "center" : void 0, alignItems: "start", children: typeof logo === "function" ? logo(animation ? false : expanded) : logo }) }),
|
|
208
|
+
/* @__PURE__ */ jsxRuntime.jsxs(ContentContainer, { children: [
|
|
209
|
+
/* @__PURE__ */ jsxRuntime.jsx(Content, { ref: contentRef, gap: 0.25, "data-is-expanded": expanded, "data-animation": animation, children }),
|
|
210
|
+
/* @__PURE__ */ jsxRuntime.jsx(StickyFooter, { "data-has-overflow-style": footerHasOverflowStyle, children: /* @__PURE__ */ jsxRuntime.jsx(ui.Tooltip, { text: expanded ? locales["navigation.collapse.button"] : locales["navigation.expand.button"], placement: "right", children: /* @__PURE__ */ jsxRuntime.jsx(ui.Button, { variant: "ghost", sentiment: "neutral", size: "small", icon: expanded ? "arrow-left-double" : "arrow-right-double", onClick: () => toggleExpand() }) }) })
|
|
211
|
+
] })
|
|
212
|
+
] }),
|
|
213
|
+
/* @__PURE__ */ jsxRuntime.jsx(Slider, { ref: sliderRef })
|
|
214
|
+
] });
|
|
215
|
+
};
|
|
216
|
+
exports.NavigationContent = NavigationContent;
|