@ultraviolet/plus 0.16.0 → 0.17.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -14,7 +14,7 @@ const StyledNav = /* @__PURE__ */ _styled("nav", process.env.NODE_ENV === "produ
14
14
  label: "StyledNav"
15
15
  })("display:flex;flex-direction:row;position:relative;border-right:1px solid ", ({
16
16
  theme
17
- }) => 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 { 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'\nimport type { NavigationProps } from './types'\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  overflow: hidden;\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\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n  onToggleExpand,\n}: NavigationProps) => {\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          onToggleExpand?.(!expanded)\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  }, [\n    expanded,\n    navigationRef,\n    onToggleExpand,\n    onWidthResize,\n    setWidth,\n    toggleExpand,\n  ])\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        {logo ? (\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        ) : null}\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={() => {\n                  toggleExpand()\n                  onToggleExpand?.(!expanded)\n                }}\n              />\n            </Tooltip>\n          </StickyFooter>\n        </ContentContainer>\n      </Container>\n      <Slider ref={sliderRef} />\n    </StyledNav>\n  )\n}\n"]} */"));
17
+ }) => 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 { 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'\nimport type { NavigationProps } from './types'\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  overflow: hidden;\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\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n  onToggleExpand,\n}: NavigationProps) => {\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    allowNavigationResize,\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          onToggleExpand?.(!expanded)\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  }, [\n    expanded,\n    navigationRef,\n    onToggleExpand,\n    onWidthResize,\n    setWidth,\n    toggleExpand,\n  ])\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        {logo ? (\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        ) : null}\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          {allowNavigationResize ? (\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={() => {\n                    toggleExpand()\n                    onToggleExpand?.(!expanded)\n                  }}\n                />\n              </Tooltip>\n            </StickyFooter>\n          ) : null}\n        </ContentContainer>\n      </Container>\n      {allowNavigationResize ? <Slider ref={sliderRef} /> : null}\n    </StyledNav>\n  )\n}\n"]} */"));
18
18
  const Container = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
19
19
  shouldForwardProp: (prop) => !["width"].includes(prop),
20
20
  target: "esezfu56"
@@ -28,7 +28,7 @@ const Container = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "produ
28
28
  width
29
29
  }) => width, "px;&[data-expanded='true'][data-animation='false']{max-width:", NAVIGATION_MAX_WIDTH, "px;min-width:", NAVIGATION_MIN_WIDTH, "px;}&[data-expanded='false']{width:", NAVIGATION_COLLASPED_WIDTH, "px;}&[data-animation='expand']{transition:width ", ANIMATION_DURATION, "ms ease-in-out;width:", ({
30
30
  width
31
- }) => width, "px;}&[data-animation='collapse']{transition:width ", ANIMATION_DURATION, "ms ease-in-out;width:", 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 { 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'\nimport type { NavigationProps } from './types'\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  overflow: hidden;\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\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n  onToggleExpand,\n}: NavigationProps) => {\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          onToggleExpand?.(!expanded)\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  }, [\n    expanded,\n    navigationRef,\n    onToggleExpand,\n    onWidthResize,\n    setWidth,\n    toggleExpand,\n  ])\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        {logo ? (\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        ) : null}\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={() => {\n                  toggleExpand()\n                  onToggleExpand?.(!expanded)\n                }}\n              />\n            </Tooltip>\n          </StickyFooter>\n        </ContentContainer>\n      </Container>\n      <Slider ref={sliderRef} />\n    </StyledNav>\n  )\n}\n"]} */"));
31
+ }) => width, "px;}&[data-animation='collapse']{transition:width ", ANIMATION_DURATION, "ms ease-in-out;width:", 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 { 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'\nimport type { NavigationProps } from './types'\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  overflow: hidden;\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\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n  onToggleExpand,\n}: NavigationProps) => {\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    allowNavigationResize,\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          onToggleExpand?.(!expanded)\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  }, [\n    expanded,\n    navigationRef,\n    onToggleExpand,\n    onWidthResize,\n    setWidth,\n    toggleExpand,\n  ])\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        {logo ? (\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        ) : null}\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          {allowNavigationResize ? (\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={() => {\n                    toggleExpand()\n                    onToggleExpand?.(!expanded)\n                  }}\n                />\n              </Tooltip>\n            </StickyFooter>\n          ) : null}\n        </ContentContainer>\n      </Container>\n      {allowNavigationResize ? <Slider ref={sliderRef} /> : null}\n    </StyledNav>\n  )\n}\n"]} */"));
32
32
  const StickyFooter = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
33
33
  target: "esezfu55"
34
34
  } : {
@@ -42,7 +42,7 @@ const StickyFooter = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "pr
42
42
  theme
43
43
  }) => `${theme.space["1"]} ${theme.space["2"]}`, ";transition:justify-content ", ANIMATION_DURATION, "ms ease-in-out;box-shadow:", ({
44
44
  theme
45
- }) => 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 { 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'\nimport type { NavigationProps } from './types'\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  overflow: hidden;\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\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n  onToggleExpand,\n}: NavigationProps) => {\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          onToggleExpand?.(!expanded)\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  }, [\n    expanded,\n    navigationRef,\n    onToggleExpand,\n    onWidthResize,\n    setWidth,\n    toggleExpand,\n  ])\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        {logo ? (\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        ) : null}\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={() => {\n                  toggleExpand()\n                  onToggleExpand?.(!expanded)\n                }}\n              />\n            </Tooltip>\n          </StickyFooter>\n        </ContentContainer>\n      </Container>\n      <Slider ref={sliderRef} />\n    </StyledNav>\n  )\n}\n"]} */"));
45
+ }) => 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 { 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'\nimport type { NavigationProps } from './types'\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  overflow: hidden;\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\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n  onToggleExpand,\n}: NavigationProps) => {\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    allowNavigationResize,\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          onToggleExpand?.(!expanded)\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  }, [\n    expanded,\n    navigationRef,\n    onToggleExpand,\n    onWidthResize,\n    setWidth,\n    toggleExpand,\n  ])\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        {logo ? (\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        ) : null}\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          {allowNavigationResize ? (\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={() => {\n                    toggleExpand()\n                    onToggleExpand?.(!expanded)\n                  }}\n                />\n              </Tooltip>\n            </StickyFooter>\n          ) : null}\n        </ContentContainer>\n      </Container>\n      {allowNavigationResize ? <Slider ref={sliderRef} /> : null}\n    </StyledNav>\n  )\n}\n"]} */"));
46
46
  const Header = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
47
47
  target: "esezfu54"
48
48
  } : {
@@ -50,7 +50,7 @@ const Header = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "producti
50
50
  label: "Header"
51
51
  })("background:", ({
52
52
  theme
53
- }) => 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 { 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'\nimport type { NavigationProps } from './types'\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  overflow: hidden;\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\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n  onToggleExpand,\n}: NavigationProps) => {\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          onToggleExpand?.(!expanded)\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  }, [\n    expanded,\n    navigationRef,\n    onToggleExpand,\n    onWidthResize,\n    setWidth,\n    toggleExpand,\n  ])\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        {logo ? (\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        ) : null}\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={() => {\n                  toggleExpand()\n                  onToggleExpand?.(!expanded)\n                }}\n              />\n            </Tooltip>\n          </StickyFooter>\n        </ContentContainer>\n      </Container>\n      <Slider ref={sliderRef} />\n    </StyledNav>\n  )\n}\n"]} */"));
53
+ }) => 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 { 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'\nimport type { NavigationProps } from './types'\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  overflow: hidden;\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\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n  onToggleExpand,\n}: NavigationProps) => {\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    allowNavigationResize,\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          onToggleExpand?.(!expanded)\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  }, [\n    expanded,\n    navigationRef,\n    onToggleExpand,\n    onWidthResize,\n    setWidth,\n    toggleExpand,\n  ])\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        {logo ? (\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        ) : null}\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          {allowNavigationResize ? (\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={() => {\n                    toggleExpand()\n                    onToggleExpand?.(!expanded)\n                  }}\n                />\n              </Tooltip>\n            </StickyFooter>\n          ) : null}\n        </ContentContainer>\n      </Container>\n      {allowNavigationResize ? <Slider ref={sliderRef} /> : null}\n    </StyledNav>\n  )\n}\n"]} */"));
54
54
  const LogoContainer = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
55
55
  target: "esezfu53"
56
56
  } : {
@@ -58,7 +58,7 @@ const LogoContainer = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "p
58
58
  label: "LogoContainer"
59
59
  })("margin:", ({
60
60
  theme
61
- }) => `${theme.space["3"]} ${theme.space["3"]} ${theme.space["2"]} ${theme.space["3"]}`, ";max-width:220px;height:22px;overflow:hidden;" + (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 { 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'\nimport type { NavigationProps } from './types'\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  overflow: hidden;\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\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n  onToggleExpand,\n}: NavigationProps) => {\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          onToggleExpand?.(!expanded)\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  }, [\n    expanded,\n    navigationRef,\n    onToggleExpand,\n    onWidthResize,\n    setWidth,\n    toggleExpand,\n  ])\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        {logo ? (\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        ) : null}\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={() => {\n                  toggleExpand()\n                  onToggleExpand?.(!expanded)\n                }}\n              />\n            </Tooltip>\n          </StickyFooter>\n        </ContentContainer>\n      </Container>\n      <Slider ref={sliderRef} />\n    </StyledNav>\n  )\n}\n"]} */"));
61
+ }) => `${theme.space["3"]} ${theme.space["3"]} ${theme.space["2"]} ${theme.space["3"]}`, ";max-width:220px;height:22px;overflow:hidden;" + (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 { 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'\nimport type { NavigationProps } from './types'\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  overflow: hidden;\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\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n  onToggleExpand,\n}: NavigationProps) => {\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    allowNavigationResize,\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          onToggleExpand?.(!expanded)\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  }, [\n    expanded,\n    navigationRef,\n    onToggleExpand,\n    onWidthResize,\n    setWidth,\n    toggleExpand,\n  ])\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        {logo ? (\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        ) : null}\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          {allowNavigationResize ? (\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={() => {\n                    toggleExpand()\n                    onToggleExpand?.(!expanded)\n                  }}\n                />\n              </Tooltip>\n            </StickyFooter>\n          ) : null}\n        </ContentContainer>\n      </Container>\n      {allowNavigationResize ? <Slider ref={sliderRef} /> : null}\n    </StyledNav>\n  )\n}\n"]} */"));
62
62
  const ContentContainer = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
63
63
  target: "esezfu52"
64
64
  } : {
@@ -70,7 +70,7 @@ const ContentContainer = /* @__PURE__ */ _styled("div", process.env.NODE_ENV ===
70
70
  } : {
71
71
  name: "12is9id",
72
72
  styles: "overflow:hidden;display:flex;flex-direction:column;flex-grow:1",
73
- 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":"AAgFmC","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 { 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'\nimport type { NavigationProps } from './types'\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  overflow: hidden;\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\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n  onToggleExpand,\n}: NavigationProps) => {\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          onToggleExpand?.(!expanded)\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  }, [\n    expanded,\n    navigationRef,\n    onToggleExpand,\n    onWidthResize,\n    setWidth,\n    toggleExpand,\n  ])\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        {logo ? (\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        ) : null}\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={() => {\n                  toggleExpand()\n                  onToggleExpand?.(!expanded)\n                }}\n              />\n            </Tooltip>\n          </StickyFooter>\n        </ContentContainer>\n      </Container>\n      <Slider ref={sliderRef} />\n    </StyledNav>\n  )\n}\n"]} */",
73
+ 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":"AAgFmC","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 { 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'\nimport type { NavigationProps } from './types'\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  overflow: hidden;\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\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n  onToggleExpand,\n}: NavigationProps) => {\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    allowNavigationResize,\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          onToggleExpand?.(!expanded)\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  }, [\n    expanded,\n    navigationRef,\n    onToggleExpand,\n    onWidthResize,\n    setWidth,\n    toggleExpand,\n  ])\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        {logo ? (\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        ) : null}\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          {allowNavigationResize ? (\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={() => {\n                    toggleExpand()\n                    onToggleExpand?.(!expanded)\n                  }}\n                />\n              </Tooltip>\n            </StickyFooter>\n          ) : null}\n        </ContentContainer>\n      </Container>\n      {allowNavigationResize ? <Slider ref={sliderRef} /> : null}\n    </StyledNav>\n  )\n}\n"]} */",
74
74
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
75
75
  });
76
76
  const Content = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
@@ -82,7 +82,7 @@ const Content = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "product
82
82
  theme
83
83
  }) => theme.space["2"], " 0;}&[data-is-expanded='true'],&[data-animation='expand']{padding:", ({
84
84
  theme
85
- }) => 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":"AAuF6B","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 { 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'\nimport type { NavigationProps } from './types'\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  overflow: hidden;\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\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n  onToggleExpand,\n}: NavigationProps) => {\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          onToggleExpand?.(!expanded)\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  }, [\n    expanded,\n    navigationRef,\n    onToggleExpand,\n    onWidthResize,\n    setWidth,\n    toggleExpand,\n  ])\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        {logo ? (\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        ) : null}\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={() => {\n                  toggleExpand()\n                  onToggleExpand?.(!expanded)\n                }}\n              />\n            </Tooltip>\n          </StickyFooter>\n        </ContentContainer>\n      </Container>\n      <Slider ref={sliderRef} />\n    </StyledNav>\n  )\n}\n"]} */"));
85
+ }) => 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":"AAuF6B","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 { 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'\nimport type { NavigationProps } from './types'\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  overflow: hidden;\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\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n  onToggleExpand,\n}: NavigationProps) => {\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    allowNavigationResize,\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          onToggleExpand?.(!expanded)\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  }, [\n    expanded,\n    navigationRef,\n    onToggleExpand,\n    onWidthResize,\n    setWidth,\n    toggleExpand,\n  ])\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        {logo ? (\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        ) : null}\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          {allowNavigationResize ? (\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={() => {\n                    toggleExpand()\n                    onToggleExpand?.(!expanded)\n                  }}\n                />\n              </Tooltip>\n            </StickyFooter>\n          ) : null}\n        </ContentContainer>\n      </Container>\n      {allowNavigationResize ? <Slider ref={sliderRef} /> : null}\n    </StyledNav>\n  )\n}\n"]} */"));
86
86
  const Slider = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
87
87
  target: "esezfu50"
88
88
  } : {
@@ -90,7 +90,7 @@ const Slider = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "producti
90
90
  label: "Slider"
91
91
  })("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:", ({
92
92
  theme
93
- }) => 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":"AAsGyB","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 { 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'\nimport type { NavigationProps } from './types'\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  overflow: hidden;\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\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n  onToggleExpand,\n}: NavigationProps) => {\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          onToggleExpand?.(!expanded)\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  }, [\n    expanded,\n    navigationRef,\n    onToggleExpand,\n    onWidthResize,\n    setWidth,\n    toggleExpand,\n  ])\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        {logo ? (\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        ) : null}\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={() => {\n                  toggleExpand()\n                  onToggleExpand?.(!expanded)\n                }}\n              />\n            </Tooltip>\n          </StickyFooter>\n        </ContentContainer>\n      </Container>\n      <Slider ref={sliderRef} />\n    </StyledNav>\n  )\n}\n"]} */"));
93
+ }) => 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":"AAsGyB","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 { 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'\nimport type { NavigationProps } from './types'\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  overflow: hidden;\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\nexport const NavigationContent = ({\n  children,\n  logo,\n  onWidthResize,\n  className,\n  id,\n  onToggleExpand,\n}: NavigationProps) => {\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    allowNavigationResize,\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          onToggleExpand?.(!expanded)\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  }, [\n    expanded,\n    navigationRef,\n    onToggleExpand,\n    onWidthResize,\n    setWidth,\n    toggleExpand,\n  ])\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        {logo ? (\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        ) : null}\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          {allowNavigationResize ? (\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={() => {\n                    toggleExpand()\n                    onToggleExpand?.(!expanded)\n                  }}\n                />\n              </Tooltip>\n            </StickyFooter>\n          ) : null}\n        </ContentContainer>\n      </Container>\n      {allowNavigationResize ? <Slider ref={sliderRef} /> : null}\n    </StyledNav>\n  )\n}\n"]} */"));
94
94
  const NavigationContent = ({
95
95
  children,
96
96
  logo,
@@ -110,7 +110,8 @@ const NavigationContent = ({
110
110
  toggleExpand,
111
111
  animation,
112
112
  locales,
113
- navigationRef
113
+ navigationRef,
114
+ allowNavigationResize
114
115
  } = context;
115
116
  const sliderRef = useRef(null);
116
117
  const contentRef = useRef(null);
@@ -205,13 +206,13 @@ const NavigationContent = ({
205
206
  logo ? /* @__PURE__ */ jsx(Header, { children: /* @__PURE__ */ jsx(LogoContainer, { justifyContent: !expanded ? "center" : void 0, alignItems: "start", children: typeof logo === "function" ? logo(animation ? false : expanded) : logo }) }) : null,
206
207
  /* @__PURE__ */ jsxs(ContentContainer, { children: [
207
208
  /* @__PURE__ */ jsx(Content, { ref: contentRef, gap: 0.25, "data-is-expanded": expanded, "data-animation": animation, children }),
208
- /* @__PURE__ */ jsx(StickyFooter, { "data-has-overflow-style": footerHasOverflowStyle, children: /* @__PURE__ */ jsx(Tooltip, { text: expanded ? locales["navigation.collapse.button"] : locales["navigation.expand.button"], placement: "right", children: /* @__PURE__ */ jsx(Button, { variant: "ghost", sentiment: "neutral", size: "small", icon: expanded ? "arrow-left-double" : "arrow-right-double", onClick: () => {
209
+ allowNavigationResize ? /* @__PURE__ */ jsx(StickyFooter, { "data-has-overflow-style": footerHasOverflowStyle, children: /* @__PURE__ */ jsx(Tooltip, { text: expanded ? locales["navigation.collapse.button"] : locales["navigation.expand.button"], placement: "right", children: /* @__PURE__ */ jsx(Button, { variant: "ghost", sentiment: "neutral", size: "small", icon: expanded ? "arrow-left-double" : "arrow-right-double", onClick: () => {
209
210
  toggleExpand();
210
211
  onToggleExpand?.(!expanded);
211
- } }) }) })
212
+ } }) }) }) : null
212
213
  ] })
213
214
  ] }),
214
- /* @__PURE__ */ jsx(Slider, { ref: sliderRef })
215
+ allowNavigationResize ? /* @__PURE__ */ jsx(Slider, { ref: sliderRef }) : null
215
216
  ] });
216
217
  };
217
218
  export {
@@ -29,6 +29,9 @@ const NavigationContext = react.createContext({
29
29
  registerItem: () => {
30
30
  },
31
31
  setPinnedItems: () => {
32
+ },
33
+ allowNavigationResize: true,
34
+ setAllowNavigationResize: () => {
32
35
  }
33
36
  });
34
37
  const useNavigation = () => react.useContext(NavigationContext);
@@ -40,12 +43,14 @@ const NavigationProvider = ({
40
43
  locales = en,
41
44
  pinLimit = 7,
42
45
  onExpandChange,
43
- initialWidth = constants.NAVIGATION_WIDTH
46
+ initialWidth = constants.NAVIGATION_WIDTH,
47
+ initialAllowNavigationResize = true
44
48
  }) => {
45
49
  const [expanded, setExpanded] = react.useState(initialExpanded);
46
50
  const [pinnedItems, setPinnedItems] = react.useState(initialPinned ?? []);
47
51
  const [animation, setAnimation] = react.useState(false);
48
52
  const [width, setWidth] = react.useState(initialWidth);
53
+ const [allowNavigationResize, setAllowNavigationResize] = react.useState(initialAllowNavigationResize);
49
54
  const [items, registerItem] = react.useReducer((oldState, newState) => ({
50
55
  ...oldState,
51
56
  ...newState
@@ -102,8 +107,10 @@ const NavigationProvider = ({
102
107
  reorderItems,
103
108
  registerItem,
104
109
  items,
105
- setPinnedItems
106
- }), [expanded, toggleExpand, pinnedItems, pinItem, unpinItem, pinnedFeature, locales, pinLimit, animation, width, reorderItems, items, setPinnedItems]);
110
+ setPinnedItems,
111
+ allowNavigationResize,
112
+ setAllowNavigationResize
113
+ }), [expanded, toggleExpand, pinnedItems, pinItem, unpinItem, pinnedFeature, locales, pinLimit, animation, width, reorderItems, items, allowNavigationResize]);
107
114
  return /* @__PURE__ */ jsxRuntime.jsx(NavigationContext.Provider, { value, children });
108
115
  };
109
116
  exports.NavigationContext = NavigationContext;