@ultraviolet/plus 0.21.28 → 0.21.29
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.
|
@@ -42,7 +42,7 @@ const RelativeDiv = /* @__PURE__ */ _styled__default.default("div", process.env.
|
|
|
42
42
|
} : {
|
|
43
43
|
name: "bjn8wh",
|
|
44
44
|
styles: "position:relative",
|
|
45
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx"],"names":[],"mappings":"AAoC8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          event.preventDefault()\n                          event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                          let newValue: string[] | undefined\n                          if (isItemPinned) {\n                            newValue = unpinItem(id)\n                          } else {\n                            newValue = pinItem(id)\n                          }\n\n                          onClickPinUnpin?.({\n                            state: isItemPinned ? 'unpin' : 'pin',\n                            id,\n                            totalPinned: newValue,\n                          })\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      event.preventDefault()\n                      event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                      let newValue: string[] | undefined\n                      if (isItemPinned) {\n                        newValue = unpinItem(id)\n                      } else {\n                        newValue = pinItem(id)\n                      }\n                      onClickPinUnpin?.({\n                        state: isItemPinned ? 'unpin' : 'pin',\n                        id,\n                        totalPinned: newValue,\n                      })\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */",
|
|
45
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx"],"names":[],"mappings":"AAoC8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          if (pinnedItems.length < pinLimit || isItemPinned) {\n                            event.preventDefault()\n                            event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                            let newValue: string[] | undefined\n                            if (isItemPinned) {\n                              newValue = unpinItem(id)\n                            } else {\n                              newValue = pinItem(id)\n                            }\n\n                            onClickPinUnpin?.({\n                              state: isItemPinned ? 'unpin' : 'pin',\n                              id,\n                              totalPinned: newValue,\n                            })\n                          }\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      if (pinnedItems.length < pinLimit || isItemPinned) {\n                        event.preventDefault()\n                        event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                        let newValue: string[] | undefined\n                        if (isItemPinned) {\n                          newValue = unpinItem(id)\n                        } else {\n                          newValue = pinItem(id)\n                        }\n                        onClickPinUnpin?.({\n                          state: isItemPinned ? 'unpin' : 'pin',\n                          id,\n                          totalPinned: newValue,\n                        })\n                      }\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */",
|
|
46
46
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
47
47
|
});
|
|
48
48
|
const StyledIcon = /* @__PURE__ */ _styled__default.default(legacy.Icon, process.env.NODE_ENV === "production" ? {
|
|
@@ -61,14 +61,14 @@ const StyledIcon = /* @__PURE__ */ _styled__default.default(legacy.Icon, process
|
|
|
61
61
|
}) => theme.colors.neutral.backgroundWeakHover, ";", ({
|
|
62
62
|
active,
|
|
63
63
|
theme
|
|
64
|
-
}) => active ? `background: ${theme.colors.primary.backgroundHover};` : null, ";}" + (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/components/Item.tsx"],"names":[],"mappings":"AA0CwB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          event.preventDefault()\n                          event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                          let newValue: string[] | undefined\n                          if (isItemPinned) {\n                            newValue = unpinItem(id)\n                          } else {\n                            newValue = pinItem(id)\n                          }\n\n                          onClickPinUnpin?.({\n                            state: isItemPinned ? 'unpin' : 'pin',\n                            id,\n                            totalPinned: newValue,\n                          })\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      event.preventDefault()\n                      event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                      let newValue: string[] | undefined\n                      if (isItemPinned) {\n                        newValue = unpinItem(id)\n                      } else {\n                        newValue = pinItem(id)\n                      }\n                      onClickPinUnpin?.({\n                        state: isItemPinned ? 'unpin' : 'pin',\n                        id,\n                        totalPinned: newValue,\n                      })\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */"));
|
|
64
|
+
}) => active ? `background: ${theme.colors.primary.backgroundHover};` : null, ";}" + (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/components/Item.tsx"],"names":[],"mappings":"AA0CwB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          if (pinnedItems.length < pinLimit || isItemPinned) {\n                            event.preventDefault()\n                            event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                            let newValue: string[] | undefined\n                            if (isItemPinned) {\n                              newValue = unpinItem(id)\n                            } else {\n                              newValue = pinItem(id)\n                            }\n\n                            onClickPinUnpin?.({\n                              state: isItemPinned ? 'unpin' : 'pin',\n                              id,\n                              totalPinned: newValue,\n                            })\n                          }\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      if (pinnedItems.length < pinLimit || isItemPinned) {\n                        event.preventDefault()\n                        event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                        let newValue: string[] | undefined\n                        if (isItemPinned) {\n                          newValue = unpinItem(id)\n                        } else {\n                          newValue = pinItem(id)\n                        }\n                        onClickPinUnpin?.({\n                          state: isItemPinned ? 'unpin' : 'pin',\n                          id,\n                          totalPinned: newValue,\n                        })\n                      }\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */"));
|
|
65
65
|
const NeutralButtonLink = process.env.NODE_ENV === "production" ? {
|
|
66
66
|
name: "rhgg7c",
|
|
67
67
|
styles: "color:inherit;text-decoration:none;background-color:inherit;border:none;text-align:left"
|
|
68
68
|
} : {
|
|
69
69
|
name: "1kb8ns1-NeutralButtonLink",
|
|
70
70
|
styles: "color:inherit;text-decoration:none;background-color:inherit;border:none;text-align:left;label:NeutralButtonLink;",
|
|
71
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx"],"names":[],"mappings":"AAyD6B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          event.preventDefault()\n                          event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                          let newValue: string[] | undefined\n                          if (isItemPinned) {\n                            newValue = unpinItem(id)\n                          } else {\n                            newValue = pinItem(id)\n                          }\n\n                          onClickPinUnpin?.({\n                            state: isItemPinned ? 'unpin' : 'pin',\n                            id,\n                            totalPinned: newValue,\n                          })\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      event.preventDefault()\n                      event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                      let newValue: string[] | undefined\n                      if (isItemPinned) {\n                        newValue = unpinItem(id)\n                      } else {\n                        newValue = pinItem(id)\n                      }\n                      onClickPinUnpin?.({\n                        state: isItemPinned ? 'unpin' : 'pin',\n                        id,\n                        totalPinned: newValue,\n                      })\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */",
|
|
71
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx"],"names":[],"mappings":"AAyD6B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          if (pinnedItems.length < pinLimit || isItemPinned) {\n                            event.preventDefault()\n                            event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                            let newValue: string[] | undefined\n                            if (isItemPinned) {\n                              newValue = unpinItem(id)\n                            } else {\n                              newValue = pinItem(id)\n                            }\n\n                            onClickPinUnpin?.({\n                              state: isItemPinned ? 'unpin' : 'pin',\n                              id,\n                              totalPinned: newValue,\n                            })\n                          }\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      if (pinnedItems.length < pinLimit || isItemPinned) {\n                        event.preventDefault()\n                        event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                        let newValue: string[] | undefined\n                        if (isItemPinned) {\n                          newValue = unpinItem(id)\n                        } else {\n                          newValue = pinItem(id)\n                        }\n                        onClickPinUnpin?.({\n                          state: isItemPinned ? 'unpin' : 'pin',\n                          id,\n                          totalPinned: newValue,\n                        })\n                      }\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */",
|
|
72
72
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
73
73
|
};
|
|
74
74
|
const LocalExpandButton = /* @__PURE__ */ _styled__default.default(ui.Button, process.env.NODE_ENV === "production" ? {
|
|
@@ -82,7 +82,7 @@ const LocalExpandButton = /* @__PURE__ */ _styled__default.default(ui.Button, pr
|
|
|
82
82
|
} : {
|
|
83
83
|
name: "9hkyle",
|
|
84
84
|
styles: "opacity:0;right:0;position:absolute;left:-24px;top:0;bottom:0;margin:auto;&:hover,&:focus,&:active{opacity:1;}",
|
|
85
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx"],"names":[],"mappings":"AAkEwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          event.preventDefault()\n                          event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                          let newValue: string[] | undefined\n                          if (isItemPinned) {\n                            newValue = unpinItem(id)\n                          } else {\n                            newValue = pinItem(id)\n                          }\n\n                          onClickPinUnpin?.({\n                            state: isItemPinned ? 'unpin' : 'pin',\n                            id,\n                            totalPinned: newValue,\n                          })\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      event.preventDefault()\n                      event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                      let newValue: string[] | undefined\n                      if (isItemPinned) {\n                        newValue = unpinItem(id)\n                      } else {\n                        newValue = pinItem(id)\n                      }\n                      onClickPinUnpin?.({\n                        state: isItemPinned ? 'unpin' : 'pin',\n                        id,\n                        totalPinned: newValue,\n                      })\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */",
|
|
85
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx"],"names":[],"mappings":"AAkEwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          if (pinnedItems.length < pinLimit || isItemPinned) {\n                            event.preventDefault()\n                            event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                            let newValue: string[] | undefined\n                            if (isItemPinned) {\n                              newValue = unpinItem(id)\n                            } else {\n                              newValue = pinItem(id)\n                            }\n\n                            onClickPinUnpin?.({\n                              state: isItemPinned ? 'unpin' : 'pin',\n                              id,\n                              totalPinned: newValue,\n                            })\n                          }\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      if (pinnedItems.length < pinLimit || isItemPinned) {\n                        event.preventDefault()\n                        event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                        let newValue: string[] | undefined\n                        if (isItemPinned) {\n                          newValue = unpinItem(id)\n                        } else {\n                          newValue = pinItem(id)\n                        }\n                        onClickPinUnpin?.({\n                          state: isItemPinned ? 'unpin' : 'pin',\n                          id,\n                          totalPinned: newValue,\n                        })\n                      }\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */",
|
|
86
86
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
87
87
|
});
|
|
88
88
|
const PinnedButton = LocalExpandButton.withComponent("div", process.env.NODE_ENV === "production" ? {
|
|
@@ -98,13 +98,13 @@ const GrabIcon = /* @__PURE__ */ _styled__default.default(legacy.Icon, process.e
|
|
|
98
98
|
label: "GrabIcon"
|
|
99
99
|
})("opacity:0;margin:0 ", ({
|
|
100
100
|
theme
|
|
101
|
-
}) => theme.space["0.25"], ";cursor:grab;" + (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/components/Item.tsx"],"names":[],"mappings":"AAoF6B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          event.preventDefault()\n                          event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                          let newValue: string[] | undefined\n                          if (isItemPinned) {\n                            newValue = unpinItem(id)\n                          } else {\n                            newValue = pinItem(id)\n                          }\n\n                          onClickPinUnpin?.({\n                            state: isItemPinned ? 'unpin' : 'pin',\n                            id,\n                            totalPinned: newValue,\n                          })\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      event.preventDefault()\n                      event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                      let newValue: string[] | undefined\n                      if (isItemPinned) {\n                        newValue = unpinItem(id)\n                      } else {\n                        newValue = pinItem(id)\n                      }\n                      onClickPinUnpin?.({\n                        state: isItemPinned ? 'unpin' : 'pin',\n                        id,\n                        totalPinned: newValue,\n                      })\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */"));
|
|
101
|
+
}) => theme.space["0.25"], ";cursor:grab;" + (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/components/Item.tsx"],"names":[],"mappings":"AAoF6B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          if (pinnedItems.length < pinLimit || isItemPinned) {\n                            event.preventDefault()\n                            event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                            let newValue: string[] | undefined\n                            if (isItemPinned) {\n                              newValue = unpinItem(id)\n                            } else {\n                              newValue = pinItem(id)\n                            }\n\n                            onClickPinUnpin?.({\n                              state: isItemPinned ? 'unpin' : 'pin',\n                              id,\n                              totalPinned: newValue,\n                            })\n                          }\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      if (pinnedItems.length < pinLimit || isItemPinned) {\n                        event.preventDefault()\n                        event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                        let newValue: string[] | undefined\n                        if (isItemPinned) {\n                          newValue = unpinItem(id)\n                        } else {\n                          newValue = pinItem(id)\n                        }\n                        onClickPinUnpin?.({\n                          state: isItemPinned ? 'unpin' : 'pin',\n                          id,\n                          totalPinned: newValue,\n                        })\n                      }\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */"));
|
|
102
102
|
const StyledBadge = /* @__PURE__ */ _styled__default.default(ui.Badge, process.env.NODE_ENV === "production" ? {
|
|
103
103
|
target: "e134hokc9"
|
|
104
104
|
} : {
|
|
105
105
|
target: "e134hokc9",
|
|
106
106
|
label: "StyledBadge"
|
|
107
|
-
})(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/components/Item.tsx"],"names":[],"mappings":"AA0FiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          event.preventDefault()\n                          event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                          let newValue: string[] | undefined\n                          if (isItemPinned) {\n                            newValue = unpinItem(id)\n                          } else {\n                            newValue = pinItem(id)\n                          }\n\n                          onClickPinUnpin?.({\n                            state: isItemPinned ? 'unpin' : 'pin',\n                            id,\n                            totalPinned: newValue,\n                          })\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      event.preventDefault()\n                      event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                      let newValue: string[] | undefined\n                      if (isItemPinned) {\n                        newValue = unpinItem(id)\n                      } else {\n                        newValue = pinItem(id)\n                      }\n                      onClickPinUnpin?.({\n                        state: isItemPinned ? 'unpin' : 'pin',\n                        id,\n                        totalPinned: newValue,\n                      })\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */");
|
|
107
|
+
})(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/components/Item.tsx"],"names":[],"mappings":"AA0FiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          if (pinnedItems.length < pinLimit || isItemPinned) {\n                            event.preventDefault()\n                            event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                            let newValue: string[] | undefined\n                            if (isItemPinned) {\n                              newValue = unpinItem(id)\n                            } else {\n                              newValue = pinItem(id)\n                            }\n\n                            onClickPinUnpin?.({\n                              state: isItemPinned ? 'unpin' : 'pin',\n                              id,\n                              totalPinned: newValue,\n                            })\n                          }\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      if (pinnedItems.length < pinLimit || isItemPinned) {\n                        event.preventDefault()\n                        event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                        let newValue: string[] | undefined\n                        if (isItemPinned) {\n                          newValue = unpinItem(id)\n                        } else {\n                          newValue = pinItem(id)\n                        }\n                        onClickPinUnpin?.({\n                          state: isItemPinned ? 'unpin' : 'pin',\n                          id,\n                          totalPinned: newValue,\n                        })\n                      }\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */");
|
|
108
108
|
const StyledMenuItem = /* @__PURE__ */ _styled__default.default(ui.MenuV2.Item, process.env.NODE_ENV === "production" ? {
|
|
109
109
|
shouldForwardProp: (prop) => !["isPinnable"].includes(prop),
|
|
110
110
|
target: "e134hokc8"
|
|
@@ -114,7 +114,7 @@ const StyledMenuItem = /* @__PURE__ */ _styled__default.default(ui.MenuV2.Item,
|
|
|
114
114
|
label: "StyledMenuItem"
|
|
115
115
|
})("text-align:left;&:hover,&:focus,&:active{", PinnedButton, "{opacity:1;}", StyledBadge, "{opacity:", ({
|
|
116
116
|
isPinnable
|
|
117
|
-
}) => isPinnable ? 0 : 1, ";}}" + (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/components/Item.tsx"],"names":[],"mappings":"AAgGE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          event.preventDefault()\n                          event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                          let newValue: string[] | undefined\n                          if (isItemPinned) {\n                            newValue = unpinItem(id)\n                          } else {\n                            newValue = pinItem(id)\n                          }\n\n                          onClickPinUnpin?.({\n                            state: isItemPinned ? 'unpin' : 'pin',\n                            id,\n                            totalPinned: newValue,\n                          })\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      event.preventDefault()\n                      event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                      let newValue: string[] | undefined\n                      if (isItemPinned) {\n                        newValue = unpinItem(id)\n                      } else {\n                        newValue = pinItem(id)\n                      }\n                      onClickPinUnpin?.({\n                        state: isItemPinned ? 'unpin' : 'pin',\n                        id,\n                        totalPinned: newValue,\n                      })\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */"));
|
|
117
|
+
}) => isPinnable ? 0 : 1, ";}}" + (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/components/Item.tsx"],"names":[],"mappings":"AAgGE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          if (pinnedItems.length < pinLimit || isItemPinned) {\n                            event.preventDefault()\n                            event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                            let newValue: string[] | undefined\n                            if (isItemPinned) {\n                              newValue = unpinItem(id)\n                            } else {\n                              newValue = pinItem(id)\n                            }\n\n                            onClickPinUnpin?.({\n                              state: isItemPinned ? 'unpin' : 'pin',\n                              id,\n                              totalPinned: newValue,\n                            })\n                          }\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      if (pinnedItems.length < pinLimit || isItemPinned) {\n                        event.preventDefault()\n                        event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                        let newValue: string[] | undefined\n                        if (isItemPinned) {\n                          newValue = unpinItem(id)\n                        } else {\n                          newValue = pinItem(id)\n                        }\n                        onClickPinUnpin?.({\n                          state: isItemPinned ? 'unpin' : 'pin',\n                          id,\n                          totalPinned: newValue,\n                        })\n                      }\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */"));
|
|
118
118
|
const StyledMenu = /* @__PURE__ */ _styled__default.default(ui.MenuV2, process.env.NODE_ENV === "production" ? {
|
|
119
119
|
target: "e134hokc7"
|
|
120
120
|
} : {
|
|
@@ -126,7 +126,7 @@ const StyledMenu = /* @__PURE__ */ _styled__default.default(ui.MenuV2, process.e
|
|
|
126
126
|
} : {
|
|
127
127
|
name: "educr3",
|
|
128
128
|
styles: "width:180px",
|
|
129
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx"],"names":[],"mappings":"AA+GiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          event.preventDefault()\n                          event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                          let newValue: string[] | undefined\n                          if (isItemPinned) {\n                            newValue = unpinItem(id)\n                          } else {\n                            newValue = pinItem(id)\n                          }\n\n                          onClickPinUnpin?.({\n                            state: isItemPinned ? 'unpin' : 'pin',\n                            id,\n                            totalPinned: newValue,\n                          })\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      event.preventDefault()\n                      event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                      let newValue: string[] | undefined\n                      if (isItemPinned) {\n                        newValue = unpinItem(id)\n                      } else {\n                        newValue = pinItem(id)\n                      }\n                      onClickPinUnpin?.({\n                        state: isItemPinned ? 'unpin' : 'pin',\n                        id,\n                        totalPinned: newValue,\n                      })\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */",
|
|
129
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx"],"names":[],"mappings":"AA+GiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          if (pinnedItems.length < pinLimit || isItemPinned) {\n                            event.preventDefault()\n                            event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                            let newValue: string[] | undefined\n                            if (isItemPinned) {\n                              newValue = unpinItem(id)\n                            } else {\n                              newValue = pinItem(id)\n                            }\n\n                            onClickPinUnpin?.({\n                              state: isItemPinned ? 'unpin' : 'pin',\n                              id,\n                              totalPinned: newValue,\n                            })\n                          }\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      if (pinnedItems.length < pinLimit || isItemPinned) {\n                        event.preventDefault()\n                        event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                        let newValue: string[] | undefined\n                        if (isItemPinned) {\n                          newValue = unpinItem(id)\n                        } else {\n                          newValue = pinItem(id)\n                        }\n                        onClickPinUnpin?.({\n                          state: isItemPinned ? 'unpin' : 'pin',\n                          id,\n                          totalPinned: newValue,\n                        })\n                      }\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */",
|
|
130
130
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
131
131
|
});
|
|
132
132
|
const PaddingStack = /* @__PURE__ */ _styled__default.default(ui.Stack, process.env.NODE_ENV === "production" ? {
|
|
@@ -140,7 +140,7 @@ const PaddingStack = /* @__PURE__ */ _styled__default.default(ui.Stack, process.
|
|
|
140
140
|
} : {
|
|
141
141
|
name: "13feash",
|
|
142
142
|
styles: "padding-left:28px",
|
|
143
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx"],"names":[],"mappings":"AAmHkC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          event.preventDefault()\n                          event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                          let newValue: string[] | undefined\n                          if (isItemPinned) {\n                            newValue = unpinItem(id)\n                          } else {\n                            newValue = pinItem(id)\n                          }\n\n                          onClickPinUnpin?.({\n                            state: isItemPinned ? 'unpin' : 'pin',\n                            id,\n                            totalPinned: newValue,\n                          })\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      event.preventDefault()\n                      event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                      let newValue: string[] | undefined\n                      if (isItemPinned) {\n                        newValue = unpinItem(id)\n                      } else {\n                        newValue = pinItem(id)\n                      }\n                      onClickPinUnpin?.({\n                        state: isItemPinned ? 'unpin' : 'pin',\n                        id,\n                        totalPinned: newValue,\n                      })\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */",
|
|
143
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx"],"names":[],"mappings":"AAmHkC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          if (pinnedItems.length < pinLimit || isItemPinned) {\n                            event.preventDefault()\n                            event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                            let newValue: string[] | undefined\n                            if (isItemPinned) {\n                              newValue = unpinItem(id)\n                            } else {\n                              newValue = pinItem(id)\n                            }\n\n                            onClickPinUnpin?.({\n                              state: isItemPinned ? 'unpin' : 'pin',\n                              id,\n                              totalPinned: newValue,\n                            })\n                          }\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      if (pinnedItems.length < pinLimit || isItemPinned) {\n                        event.preventDefault()\n                        event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                        let newValue: string[] | undefined\n                        if (isItemPinned) {\n                          newValue = unpinItem(id)\n                        } else {\n                          newValue = pinItem(id)\n                        }\n                        onClickPinUnpin?.({\n                          state: isItemPinned ? 'unpin' : 'pin',\n                          id,\n                          totalPinned: newValue,\n                        })\n                      }\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */",
|
|
144
144
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
145
145
|
});
|
|
146
146
|
const AnimatedIcon = /* @__PURE__ */ _styled__default.default(legacy.Icon, process.env.NODE_ENV === "production" ? {
|
|
@@ -148,7 +148,7 @@ const AnimatedIcon = /* @__PURE__ */ _styled__default.default(legacy.Icon, proce
|
|
|
148
148
|
} : {
|
|
149
149
|
target: "e134hokc5",
|
|
150
150
|
label: "AnimatedIcon"
|
|
151
|
-
})(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/components/Item.tsx"],"names":[],"mappings":"AAuHiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          event.preventDefault()\n                          event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                          let newValue: string[] | undefined\n                          if (isItemPinned) {\n                            newValue = unpinItem(id)\n                          } else {\n                            newValue = pinItem(id)\n                          }\n\n                          onClickPinUnpin?.({\n                            state: isItemPinned ? 'unpin' : 'pin',\n                            id,\n                            totalPinned: newValue,\n                          })\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      event.preventDefault()\n                      event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                      let newValue: string[] | undefined\n                      if (isItemPinned) {\n                        newValue = unpinItem(id)\n                      } else {\n                        newValue = pinItem(id)\n                      }\n                      onClickPinUnpin?.({\n                        state: isItemPinned ? 'unpin' : 'pin',\n                        id,\n                        totalPinned: newValue,\n                      })\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */");
|
|
151
|
+
})(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/components/Item.tsx"],"names":[],"mappings":"AAuHiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          if (pinnedItems.length < pinLimit || isItemPinned) {\n                            event.preventDefault()\n                            event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                            let newValue: string[] | undefined\n                            if (isItemPinned) {\n                              newValue = unpinItem(id)\n                            } else {\n                              newValue = pinItem(id)\n                            }\n\n                            onClickPinUnpin?.({\n                              state: isItemPinned ? 'unpin' : 'pin',\n                              id,\n                              totalPinned: newValue,\n                            })\n                          }\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      if (pinnedItems.length < pinLimit || isItemPinned) {\n                        event.preventDefault()\n                        event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                        let newValue: string[] | undefined\n                        if (isItemPinned) {\n                          newValue = unpinItem(id)\n                        } else {\n                          newValue = pinItem(id)\n                        }\n                        onClickPinUnpin?.({\n                          state: isItemPinned ? 'unpin' : 'pin',\n                          id,\n                          totalPinned: newValue,\n                        })\n                      }\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */");
|
|
152
152
|
const WrapText = /* @__PURE__ */ _styled__default.default(ui.Text, process.env.NODE_ENV === "production" ? {
|
|
153
153
|
shouldForwardProp: (prop) => !["animation", "subLabel", "textProminence"].includes(prop),
|
|
154
154
|
target: "e134hokc4"
|
|
@@ -160,7 +160,7 @@ const WrapText = /* @__PURE__ */ _styled__default.default(ui.Text, process.env.N
|
|
|
160
160
|
animation
|
|
161
161
|
}) => animation ? "normal" : "anywhere", ";white-space:", ({
|
|
162
162
|
animation
|
|
163
|
-
}) => animation ? "nowrap" : "normal", ";overflow:hidden;display:-webkit-box;-webkit-box-orient:vertical;-webkit-line-clamp:2;white-space:pre-wrap;" + (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/components/Item.tsx"],"names":[],"mappings":"AA+HE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          event.preventDefault()\n                          event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                          let newValue: string[] | undefined\n                          if (isItemPinned) {\n                            newValue = unpinItem(id)\n                          } else {\n                            newValue = pinItem(id)\n                          }\n\n                          onClickPinUnpin?.({\n                            state: isItemPinned ? 'unpin' : 'pin',\n                            id,\n                            totalPinned: newValue,\n                          })\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      event.preventDefault()\n                      event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                      let newValue: string[] | undefined\n                      if (isItemPinned) {\n                        newValue = unpinItem(id)\n                      } else {\n                        newValue = pinItem(id)\n                      }\n                      onClickPinUnpin?.({\n                        state: isItemPinned ? 'unpin' : 'pin',\n                        id,\n                        totalPinned: newValue,\n                      })\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */"));
|
|
163
|
+
}) => animation ? "nowrap" : "normal", ";overflow:hidden;display:-webkit-box;-webkit-box-orient:vertical;-webkit-line-clamp:2;white-space:pre-wrap;" + (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/components/Item.tsx"],"names":[],"mappings":"AA+HE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          if (pinnedItems.length < pinLimit || isItemPinned) {\n                            event.preventDefault()\n                            event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                            let newValue: string[] | undefined\n                            if (isItemPinned) {\n                              newValue = unpinItem(id)\n                            } else {\n                              newValue = pinItem(id)\n                            }\n\n                            onClickPinUnpin?.({\n                              state: isItemPinned ? 'unpin' : 'pin',\n                              id,\n                              totalPinned: newValue,\n                            })\n                          }\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      if (pinnedItems.length < pinLimit || isItemPinned) {\n                        event.preventDefault()\n                        event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                        let newValue: string[] | undefined\n                        if (isItemPinned) {\n                          newValue = unpinItem(id)\n                        } else {\n                          newValue = pinItem(id)\n                        }\n                        onClickPinUnpin?.({\n                          state: isItemPinned ? 'unpin' : 'pin',\n                          id,\n                          totalPinned: newValue,\n                        })\n                      }\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */"));
|
|
164
164
|
const StyledStack = /* @__PURE__ */ _styled__default.default(ui.Stack, process.env.NODE_ENV === "production" ? {
|
|
165
165
|
target: "e134hokc3"
|
|
166
166
|
} : {
|
|
@@ -172,7 +172,7 @@ const StyledStack = /* @__PURE__ */ _styled__default.default(ui.Stack, process.e
|
|
|
172
172
|
} : {
|
|
173
173
|
name: "13feash",
|
|
174
174
|
styles: "padding-left:28px",
|
|
175
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx"],"names":[],"mappings":"AAyIiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          event.preventDefault()\n                          event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                          let newValue: string[] | undefined\n                          if (isItemPinned) {\n                            newValue = unpinItem(id)\n                          } else {\n                            newValue = pinItem(id)\n                          }\n\n                          onClickPinUnpin?.({\n                            state: isItemPinned ? 'unpin' : 'pin',\n                            id,\n                            totalPinned: newValue,\n                          })\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      event.preventDefault()\n                      event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                      let newValue: string[] | undefined\n                      if (isItemPinned) {\n                        newValue = unpinItem(id)\n                      } else {\n                        newValue = pinItem(id)\n                      }\n                      onClickPinUnpin?.({\n                        state: isItemPinned ? 'unpin' : 'pin',\n                        id,\n                        totalPinned: newValue,\n                      })\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */",
|
|
175
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx"],"names":[],"mappings":"AAyIiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          if (pinnedItems.length < pinLimit || isItemPinned) {\n                            event.preventDefault()\n                            event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                            let newValue: string[] | undefined\n                            if (isItemPinned) {\n                              newValue = unpinItem(id)\n                            } else {\n                              newValue = pinItem(id)\n                            }\n\n                            onClickPinUnpin?.({\n                              state: isItemPinned ? 'unpin' : 'pin',\n                              id,\n                              totalPinned: newValue,\n                            })\n                          }\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      if (pinnedItems.length < pinLimit || isItemPinned) {\n                        event.preventDefault()\n                        event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                        let newValue: string[] | undefined\n                        if (isItemPinned) {\n                          newValue = unpinItem(id)\n                        } else {\n                          newValue = pinItem(id)\n                        }\n                        onClickPinUnpin?.({\n                          state: isItemPinned ? 'unpin' : 'pin',\n                          id,\n                          totalPinned: newValue,\n                        })\n                      }\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */",
|
|
176
176
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
177
177
|
});
|
|
178
178
|
const StyledContainer = /* @__PURE__ */ _styled__default.default(ui.Stack, process.env.NODE_ENV === "production" ? {
|
|
@@ -204,7 +204,7 @@ const StyledContainer = /* @__PURE__ */ _styled__default.default(ui.Stack, proce
|
|
|
204
204
|
theme
|
|
205
205
|
}) => theme.colors.primary.backgroundHover, ";}}&[disabled]{cursor:not-allowed;background-color:unset;", WrapText, "{color:", ({
|
|
206
206
|
theme
|
|
207
|
-
}) => theme.colors.neutral.textWeakDisabled, ';}}&[data-animation="collapse"][data-animation-type="complex"]{animation:', constants.shrinkHeight, " ", constants.ANIMATION_DURATION, "ms ease-in-out;", WrapText, ",", AnimatedIcon, ",", StyledBadge, "{animation:", ui.fadeIn, " ", constants.ANIMATION_DURATION, 'ms ease-in-out reverse;}}&[data-animation="expand"][data-animation-type="complex"]{animation:', constants.shrinkHeight, " ", constants.ANIMATION_DURATION, "ms ease-in-out reverse;", WrapText, ",", AnimatedIcon, ",", StyledBadge, "{animation:", ui.fadeIn, " ", constants.ANIMATION_DURATION, "ms ease-in-out;}", StyledStack, "{display: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/components/Item.tsx"],"names":[],"mappings":"AA6IqC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          event.preventDefault()\n                          event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                          let newValue: string[] | undefined\n                          if (isItemPinned) {\n                            newValue = unpinItem(id)\n                          } else {\n                            newValue = pinItem(id)\n                          }\n\n                          onClickPinUnpin?.({\n                            state: isItemPinned ? 'unpin' : 'pin',\n                            id,\n                            totalPinned: newValue,\n                          })\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      event.preventDefault()\n                      event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                      let newValue: string[] | undefined\n                      if (isItemPinned) {\n                        newValue = unpinItem(id)\n                      } else {\n                        newValue = pinItem(id)\n                      }\n                      onClickPinUnpin?.({\n                        state: isItemPinned ? 'unpin' : 'pin',\n                        id,\n                        totalPinned: newValue,\n                      })\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */"));
|
|
207
|
+
}) => theme.colors.neutral.textWeakDisabled, ';}}&[data-animation="collapse"][data-animation-type="complex"]{animation:', constants.shrinkHeight, " ", constants.ANIMATION_DURATION, "ms ease-in-out;", WrapText, ",", AnimatedIcon, ",", StyledBadge, "{animation:", ui.fadeIn, " ", constants.ANIMATION_DURATION, 'ms ease-in-out reverse;}}&[data-animation="expand"][data-animation-type="complex"]{animation:', constants.shrinkHeight, " ", constants.ANIMATION_DURATION, "ms ease-in-out reverse;", WrapText, ",", AnimatedIcon, ",", StyledBadge, "{animation:", ui.fadeIn, " ", constants.ANIMATION_DURATION, "ms ease-in-out;}", StyledStack, "{display: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/components/Item.tsx"],"names":[],"mappings":"AA6IqC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          if (pinnedItems.length < pinLimit || isItemPinned) {\n                            event.preventDefault()\n                            event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                            let newValue: string[] | undefined\n                            if (isItemPinned) {\n                              newValue = unpinItem(id)\n                            } else {\n                              newValue = pinItem(id)\n                            }\n\n                            onClickPinUnpin?.({\n                              state: isItemPinned ? 'unpin' : 'pin',\n                              id,\n                              totalPinned: newValue,\n                            })\n                          }\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      if (pinnedItems.length < pinLimit || isItemPinned) {\n                        event.preventDefault()\n                        event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                        let newValue: string[] | undefined\n                        if (isItemPinned) {\n                          newValue = unpinItem(id)\n                        } else {\n                          newValue = pinItem(id)\n                        }\n                        onClickPinUnpin?.({\n                          state: isItemPinned ? 'unpin' : 'pin',\n                          id,\n                          totalPinned: newValue,\n                        })\n                      }\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */"));
|
|
208
208
|
const MenuStack = /* @__PURE__ */ _styled__default.default(ui.Stack, process.env.NODE_ENV === "production" ? {
|
|
209
209
|
target: "e134hokc1"
|
|
210
210
|
} : {
|
|
@@ -214,7 +214,7 @@ const MenuStack = /* @__PURE__ */ _styled__default.default(ui.Stack, process.env
|
|
|
214
214
|
theme
|
|
215
215
|
}) => `0 ${theme.space["2"]}`, ";margin-top:", ({
|
|
216
216
|
theme
|
|
217
|
-
}) => theme.space["0.25"], ";" + (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/components/Item.tsx"],"names":[],"mappings":"AAwP+B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          event.preventDefault()\n                          event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                          let newValue: string[] | undefined\n                          if (isItemPinned) {\n                            newValue = unpinItem(id)\n                          } else {\n                            newValue = pinItem(id)\n                          }\n\n                          onClickPinUnpin?.({\n                            state: isItemPinned ? 'unpin' : 'pin',\n                            id,\n                            totalPinned: newValue,\n                          })\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      event.preventDefault()\n                      event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                      let newValue: string[] | undefined\n                      if (isItemPinned) {\n                        newValue = unpinItem(id)\n                      } else {\n                        newValue = pinItem(id)\n                      }\n                      onClickPinUnpin?.({\n                        state: isItemPinned ? 'unpin' : 'pin',\n                        id,\n                        totalPinned: newValue,\n                      })\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */"));
|
|
217
|
+
}) => theme.space["0.25"], ";" + (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/components/Item.tsx"],"names":[],"mappings":"AAwP+B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          if (pinnedItems.length < pinLimit || isItemPinned) {\n                            event.preventDefault()\n                            event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                            let newValue: string[] | undefined\n                            if (isItemPinned) {\n                              newValue = unpinItem(id)\n                            } else {\n                              newValue = pinItem(id)\n                            }\n\n                            onClickPinUnpin?.({\n                              state: isItemPinned ? 'unpin' : 'pin',\n                              id,\n                              totalPinned: newValue,\n                            })\n                          }\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      if (pinnedItems.length < pinLimit || isItemPinned) {\n                        event.preventDefault()\n                        event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                        let newValue: string[] | undefined\n                        if (isItemPinned) {\n                          newValue = unpinItem(id)\n                        } else {\n                          newValue = pinItem(id)\n                        }\n                        onClickPinUnpin?.({\n                          state: isItemPinned ? 'unpin' : 'pin',\n                          id,\n                          totalPinned: newValue,\n                        })\n                      }\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */"));
|
|
218
218
|
const ContainerCategoryIcon = /* @__PURE__ */ _styled__default.default(ui.Stack, process.env.NODE_ENV === "production" ? {
|
|
219
219
|
target: "e134hokc0"
|
|
220
220
|
} : {
|
|
@@ -226,7 +226,7 @@ const ContainerCategoryIcon = /* @__PURE__ */ _styled__default.default(ui.Stack,
|
|
|
226
226
|
} : {
|
|
227
227
|
name: "d47oax",
|
|
228
228
|
styles: "min-width:20px",
|
|
229
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx"],"names":[],"mappings":"AA6P2C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          event.preventDefault()\n                          event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                          let newValue: string[] | undefined\n                          if (isItemPinned) {\n                            newValue = unpinItem(id)\n                          } else {\n                            newValue = pinItem(id)\n                          }\n\n                          onClickPinUnpin?.({\n                            state: isItemPinned ? 'unpin' : 'pin',\n                            id,\n                            totalPinned: newValue,\n                          })\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      event.preventDefault()\n                      event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                      let newValue: string[] | undefined\n                      if (isItemPinned) {\n                        newValue = unpinItem(id)\n                      } else {\n                        newValue = pinItem(id)\n                      }\n                      onClickPinUnpin?.({\n                        state: isItemPinned ? 'unpin' : 'pin',\n                        id,\n                        totalPinned: newValue,\n                      })\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */",
|
|
229
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx"],"names":[],"mappings":"AA6P2C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/Navigation/components/Item.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport * as CategoryIcon from '@ultraviolet/icons/category'\nimport { ConsoleCategoryIcon } from '@ultraviolet/icons/category'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport {\n  Badge,\n  Button,\n  Expandable,\n  MenuV2,\n  Stack,\n  Text,\n  Tooltip,\n  fadeIn,\n} from '@ultraviolet/ui'\nimport type {\n  ComponentProps,\n  DragEvent,\n  JSX,\n  MouseEvent,\n  ReactNode,\n} from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useMemo,\n  useReducer,\n} from 'react'\nimport type { PascalToCamelCaseWithoutSuffix } from '../../../types'\nimport { useNavigation } from '../NavigationProvider'\nimport { ANIMATION_DURATION, shrinkHeight } from '../constants'\nimport type { PinUnPinType } from '../types'\n\nconst RelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledIcon = styled(Icon, {\n  shouldForwardProp: prop => !['active'].includes(prop),\n})<{ active?: boolean }>`\n  position: absolute;\n  top: 0;\n  bottom: 0;\n  margin: auto 0;\n  padding: ${({ theme }) => theme.space['0.25']};\n  border-radius: ${({ theme }) => theme.radii.default};\n  &:hover {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n\n    ${({ active, theme }) =>\n      active ? `background: ${theme.colors.primary.backgroundHover};` : null}\n  }\n`\n\nconst NeutralButtonLink = css`\n  color: inherit;\n  text-decoration: none;\n  background-color: inherit;\n  border: none;\n  text-align: left;\n`\n\n// Pin button when the navigation is expanded\nconst LocalExpandButton = styled(Button)`\n  opacity: 0;\n  right: 0;\n  position: absolute;\n  left: -24px;\n  top: 0;\n  bottom: 0;\n  margin: auto;\n\n  &:hover,\n  &:focus,\n  &:active {\n    opacity: 1;\n  }\n`\n\nconst PinnedButton = LocalExpandButton.withComponent('div')\n\nconst GrabIcon = styled(Icon)`\n  opacity: 0;\n  margin: 0 ${({ theme }) => theme.space['0.25']};\n  cursor: grab;\n`\n\nconst StyledBadge = styled(Badge)``\n\nconst StyledMenuItem = styled(MenuV2.Item, {\n  shouldForwardProp: prop => !['isPinnable'].includes(prop),\n})<{\n  isPinnable?: boolean\n}>`\n  text-align: left;\n  &:hover,\n  &:focus,\n  &:active {\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    ${StyledBadge} {\n      opacity: ${({ isPinnable }) => (isPinnable ? 0 : 1)};\n    }\n  }\n`\n\nconst StyledMenu = styled(MenuV2)`\n  width: 180px;\n`\n\nconst PaddingStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst AnimatedIcon = styled(Icon)``\n\nconst WrapText = styled(Text, {\n  shouldForwardProp: prop =>\n    !['animation', 'subLabel', 'textProminence'].includes(prop),\n})<{\n  animation?: 'collapse' | 'expand' | boolean\n  subLabel?: boolean\n}>`\n  overflow-wrap: ${({ animation }) => (animation ? 'normal' : 'anywhere')};\n  white-space: ${({ animation }) => (animation ? 'nowrap' : 'normal')};\n  overflow: hidden;\n  display: -webkit-box;\n  -webkit-box-orient: vertical;\n  -webkit-line-clamp: 2;\n  white-space: pre-wrap;\n`\n\nconst StyledStack = styled(Stack)`\n  padding-left: 28px; // This value needs to be hardcoded because of the category icon size\n`\n\nconst StyledContainer = styled(Stack)`\n  ${NeutralButtonLink};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-has-no-expand=\"false\"] {\n    cursor: pointer;\n  }\n  margin-top: ${({ theme }) => theme.space['0.25']};\n  padding: ${({ theme }) =>\n    `calc(${theme.space['0.25']} + ${theme.space['0.5']}) ${theme.space['1']}`};\n\n  &[data-has-sub-label=\"true\"] {\n    padding: ${({ theme }) => `${theme.space['0.5']} ${theme.space['1']}`};\n  }\n\n  width: 100%;\n\n  &:hover[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ),\n  &:focus[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n  &[data-has-active-children=\"true\"][data-has-no-expand=\"false\"]:not(\n      [disabled][data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeakHover};\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n\n    ${PinnedButton} {\n      opacity: 1;\n    }\n\n    &[data-is-pinnable=\"true\"] {\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &[data-has-no-expand=\"false\"]:not([disabled]) {\n    &:hover,\n    &:focus,\n    &:active {\n      ${PinnedButton}, ${GrabIcon} {\n        opacity: 1;\n      }\n\n      ${StyledBadge} {\n        opacity: 0;\n      }\n    }\n  }\n\n  &:hover[data-has-children=\"false\"][data-is-active=\"false\"]:not([disabled]) {\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakHover};\n    }\n  }\n\n  &:active[data-has-no-expand=\"false\"]:not([disabled]):not(\n      [data-is-active=\"true\"]\n    ) {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n\n  &[data-is-active=\"true\"],\n  &:hover[data-has-active=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n\n    &:hover {\n      background-color: ${({ theme }) => theme.colors.primary.backgroundHover};\n    }\n  }\n\n  &[disabled] {\n    cursor: not-allowed;\n    background-color: unset;\n\n    ${WrapText} {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &[data-animation=\"collapse\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    }\n  }\n\n  &[data-animation=\"expand\"][data-animation-type=\"complex\"] {\n    animation: ${shrinkHeight} ${ANIMATION_DURATION}ms ease-in-out reverse;\n    ${WrapText}, ${AnimatedIcon}, ${StyledBadge} {\n      animation: ${fadeIn} ${ANIMATION_DURATION}ms ease-in-out;\n    }\n\n    ${StyledStack} {\n      display: none;\n    }\n  }\n`\n\nconst MenuStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  margin-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst ContainerCategoryIcon = styled(Stack)`\n  min-width: 20px;\n`\n\ntype ItemType = 'default' | 'pinned' | 'pinnedGroup'\n\ntype ItemProps = {\n  children?: ReactNode\n  /**\n   * Sets a category icon on the left of the item\n   */\n  categoryIcon?: PascalToCamelCaseWithoutSuffix<\n    keyof typeof CategoryIcon,\n    'CategoryIcon'\n  >\n  categoryIconVariant?: ComponentProps<\n    (typeof CategoryIcon)['BaremetalCategoryIcon']\n  >['variant']\n  /**\n   * The label of the item that will be shown.\n   * It is also used as the key for pinning.\n   */\n  label: string\n  /**\n   * It should be a unique id and will be used for pin/unpin feature.\n   */\n  id: string\n  /**\n   * Text shown under the label with a lighter color and smaller font size\n   */\n  subLabel?: string\n  /**\n   * Badge is added on the right of the item. It is hidden on hover if pinned\n   * feature is enabled\n   */\n  badgeText?: string\n  /**\n   * Defined the sentiment of the badge according to Badge component from\n   * `@ultraviolet/ui`\n   */\n  badgeSentiment?: ComponentProps<typeof Badge>['sentiment']\n  href?: string\n  /**\n   * This function will be triggered on click of the item. If the item is expandable\n   * toggle will be passed with it.\n   */\n  onToggle?: (toggle: boolean) => void\n  onClickPinUnpin?: (parameters: PinUnPinType) => void\n  /**\n   * This prop is used to control if the item is expanded or collapsed\n   */\n  toggle?: boolean\n  /**\n   * Set this to true if your current page is this item.\n   */\n  active?: boolean\n  /**\n   * If you want to remove pin button on your item use this prop\n   */\n  noPinButton?: boolean\n  /**\n   * You don't need to use this prop it's used internally to control the type of the item\n   */\n  type?: ItemType\n  /**\n   * You don't need to use this prop it's used internally to control if the item has a parent\n   */\n  hasParents?: boolean\n  /**\n   * You don't need to use this prop it's used internally for pinned item to be reorganized with drag and drop\n   */\n  index?: number\n  /**\n   * When the item has href it becomes a link if not it is a button.\n   * When using an external routing tool you might need to remove both of them and use\n   * a non focusable element. This option allows you to choose the tag of the\n   * item.\n   */\n  as?: keyof JSX.IntrinsicElements\n  /**\n   * Use this prop if you want to remove the expand behavior when the item\n   * has sub items.\n   */\n  noExpand?: boolean\n  disabled?: boolean\n  'data-testid'?: string\n}\n\nexport const Item = ({\n  children,\n  categoryIcon,\n  categoryIconVariant,\n  label,\n  subLabel,\n  badgeText,\n  badgeSentiment,\n  href,\n  onToggle,\n  onClickPinUnpin,\n  toggle,\n  active,\n  noPinButton,\n  type = 'default',\n  hasParents,\n  as,\n  disabled,\n  noExpand = false,\n  index,\n  id,\n  'data-testid': dataTestId,\n}: ItemProps) => {\n  const context = useNavigation()\n  if (!context) {\n    throw new Error(\n      'Navigation.Item can only be used inside a NavigationProvider.',\n    )\n  }\n\n  const {\n    expanded,\n    locales,\n    pinnedFeature,\n    pinItem,\n    unpinItem,\n    pinnedItems,\n    pinLimit,\n    animation,\n    registerItem,\n    shouldAnimate,\n    animationType,\n  } = context\n\n  useEffect(\n    () => {\n      if (type !== 'pinnedGroup') {\n        registerItem({ [id]: { label, active, onToggle, onClickPinUnpin } })\n      }\n    },\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    [active, id, label, registerItem],\n  )\n\n  const [internalExpanded, onToggleExpand] = useReducer(\n    prevState => !prevState,\n    Boolean(toggle),\n  )\n\n  const triggerToggle = useCallback(() => {\n    onToggleExpand()\n    onToggle?.(internalExpanded)\n  }, [internalExpanded, onToggle])\n\n  const PaddedStack = noExpand || type === 'pinnedGroup' ? Stack : PaddingStack\n\n  const hasHrefAndNoChildren = href && !children\n  const hasPinnedFeatureAndNoChildren =\n    pinnedFeature && !children && !noPinButton\n  const isItemPinned = pinnedItems.includes(id)\n  const shouldShowPinnedButton = useMemo(() => {\n    if (href || disabled) return false\n\n    if (hasPinnedFeatureAndNoChildren && type !== 'default') {\n      return true\n    }\n\n    if (hasPinnedFeatureAndNoChildren) {\n      return true\n    }\n\n    return false\n  }, [disabled, hasPinnedFeatureAndNoChildren, href, type])\n\n  const hasActiveChildren = useMemo(() => {\n    if (!children) return false\n\n    return (\n      Children.map(children, child =>\n        isValidElement<ItemProps>(child) ? child.props?.active : false,\n      ) as boolean[]\n    ).includes(true)\n  }, [children])\n\n  const containerTag = useMemo(() => {\n    if (as) {\n      return as\n    }\n\n    if (hasHrefAndNoChildren) {\n      return 'a'\n    }\n\n    if (noExpand) {\n      return 'div'\n    }\n\n    return 'button'\n  }, [as, hasHrefAndNoChildren, noExpand])\n\n  const Container = useMemo(\n    () => StyledContainer.withComponent(containerTag),\n    [containerTag],\n  )\n\n  const CategoryIconUsed = categoryIcon\n    ? CategoryIcon[\n        `${\n          categoryIcon.charAt(0).toUpperCase() + categoryIcon.slice(1)\n        }CategoryIcon` as keyof typeof CategoryIcon\n      ]\n    : null\n\n  const ariaExpanded = useMemo(() => {\n    if (hasHrefAndNoChildren && internalExpanded) {\n      return true\n    }\n\n    if (hasHrefAndNoChildren && !internalExpanded) {\n      return false\n    }\n\n    return undefined\n  }, [hasHrefAndNoChildren, internalExpanded])\n\n  const isPinDisabled = pinnedItems.length >= pinLimit\n  const pinTooltipLocale = useMemo(() => {\n    if (isPinDisabled) {\n      return locales['navigation.pin.limit']\n    }\n\n    if (isItemPinned) {\n      return locales['navigation.unpin.tooltip']\n    }\n\n    return locales['navigation.pin.tooltip']\n  }, [isItemPinned, isPinDisabled, locales])\n\n  const onDragStartTrigger = (event: DragEvent<HTMLDivElement>) => {\n    event.dataTransfer.setData('text/plain', JSON.stringify({ label, index }))\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '0.5'\n  }\n\n  const onDragStopTrigger = (event: DragEvent<HTMLDivElement>) => {\n    // eslint-disable-next-line no-param-reassign\n    event.currentTarget.style.opacity = '1'\n  }\n\n  const expandableAnimationDuration = useMemo(() => {\n    if (!shouldAnimate || animationType === 'simple') return 0\n\n    // Avoid animation of all expendable Item during collapse, expend of the Navigation\n    if (shouldAnimate && typeof animation !== 'string') {\n      return ANIMATION_DURATION\n    }\n\n    return 0\n  }, [animation, shouldAnimate, animationType])\n\n  // This content is when the navigation is expanded\n  if (expanded || (!expanded && animation === 'expand')) {\n    const renderChildren = Children.map(children, child =>\n      isValidElement<ItemProps>(child)\n        ? cloneElement(child, {\n            hasParents: true,\n          })\n        : child,\n    )\n\n    return (\n      <>\n        <Container\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          data-has-sub-label={!!subLabel}\n          onClick={triggerToggle}\n          aria-expanded={ariaExpanded}\n          href={href}\n          target={href ? '_blank' : undefined}\n          data-is-pinnable={shouldShowPinnedButton}\n          data-is-active={active}\n          data-animation={shouldAnimate ? animation : undefined}\n          data-animation-type={animationType}\n          data-has-children={!!children}\n          data-has-active-children={hasActiveChildren}\n          data-has-no-expand={noExpand}\n          disabled={disabled}\n          draggable={type === 'pinned' && expanded}\n          onDragStart={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStartTrigger(event) : undefined\n          }\n          onDragEnd={(event: DragEvent<HTMLDivElement>) =>\n            expanded ? onDragStopTrigger(event) : undefined\n          }\n          id={id}\n          data-testId={dataTestId}\n        >\n          <Stack\n            direction=\"row\"\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n          >\n            {CategoryIconUsed ? (\n              <ContainerCategoryIcon\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                <CategoryIconUsed\n                  variant={active ? 'primary' : categoryIconVariant}\n                  disabled={disabled}\n                />\n              </ContainerCategoryIcon>\n            ) : null}\n            {type === 'pinned' && expanded ? (\n              <GrabIcon\n                name=\"drag-vertical\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                size=\"small\"\n                disabled={disabled}\n              />\n            ) : null}\n            <Stack>\n              <WrapText\n                as=\"span\"\n                variant=\"bodySmallStrong\"\n                sentiment={active ? 'primary' : 'neutral'}\n                prominence={\n                  (categoryIcon || !hasParents) && !active\n                    ? 'strong'\n                    : 'default'\n                }\n                animation={animation}\n                disabled={disabled}\n              >\n                {label}\n              </WrapText>\n              {subLabel ? (\n                <WrapText\n                  as=\"span\"\n                  variant=\"caption\"\n                  sentiment=\"neutral\"\n                  prominence=\"weak\"\n                  animation={animation}\n                  disabled={disabled}\n                  subLabel\n                >\n                  {subLabel}\n                </WrapText>\n              ) : null}\n            </Stack>\n          </Stack>\n          <Stack direction=\"row\" alignItems=\"center\" gap={href ? 1 : undefined}>\n            {badgeText || hasPinnedFeatureAndNoChildren ? (\n              <>\n                {badgeText ? (\n                  <StyledBadge\n                    sentiment={badgeSentiment}\n                    size=\"small\"\n                    prominence=\"strong\"\n                    disabled={disabled}\n                  >\n                    {badgeText}\n                  </StyledBadge>\n                ) : null}\n                {shouldShowPinnedButton ? (\n                  <Tooltip\n                    text={\n                      isItemPinned\n                        ? locales['navigation.unpin.tooltip']\n                        : pinTooltipLocale\n                    }\n                    placement=\"right\"\n                  >\n                    <RelativeDiv>\n                      <PinnedButton\n                        role=\"button\"\n                        aria-label={isItemPinned ? 'unpin' : 'pin'}\n                        size=\"xsmall\"\n                        variant=\"ghost\"\n                        sentiment={active ? 'primary' : 'neutral'}\n                        onClick={(event: MouseEvent<HTMLDivElement>) => {\n                          if (pinnedItems.length < pinLimit || isItemPinned) {\n                            event.preventDefault()\n                            event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n                            let newValue: string[] | undefined\n                            if (isItemPinned) {\n                              newValue = unpinItem(id)\n                            } else {\n                              newValue = pinItem(id)\n                            }\n\n                            onClickPinUnpin?.({\n                              state: isItemPinned ? 'unpin' : 'pin',\n                              id,\n                              totalPinned: newValue,\n                            })\n                          }\n                        }}\n                        disabled={isItemPinned ? false : isPinDisabled}\n                      >\n                        <StyledIcon\n                          size=\"large\"\n                          name={isItemPinned ? 'unpin' : 'pin'}\n                          variant={isItemPinned ? 'filled' : 'outlined'}\n                          disabled={isItemPinned ? false : isPinDisabled}\n                          sentiment={active ? 'primary' : 'neutral'}\n                          active={active}\n                        />\n                      </PinnedButton>\n                    </RelativeDiv>\n                  </Tooltip>\n                ) : null}\n              </>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"default\"\n                disabled={disabled}\n              />\n            ) : null}\n            {children ? (\n              <Stack gap={1} direction=\"row\" alignItems=\"center\">\n                {!animation && !noExpand ? (\n                  <AnimatedIcon\n                    name={internalExpanded ? 'arrow-down' : 'arrow-right'}\n                    sentiment=\"neutral\"\n                    prominence=\"weak\"\n                  />\n                ) : null}\n              </Stack>\n            ) : null}\n          </Stack>\n        </Container>\n        {children ? (\n          <>\n            {!noExpand ? (\n              <Expandable\n                opened={internalExpanded}\n                animationDuration={expandableAnimationDuration}\n              >\n                <PaddedStack>{renderChildren}</PaddedStack>\n              </Expandable>\n            ) : (\n              <PaddedStack>{renderChildren}</PaddedStack>\n            )}\n          </>\n        ) : null}\n      </>\n    )\n  }\n\n  // This content is the menu of the navigation when collapsed\n  if (categoryIcon || (Children.count(children) > 0 && !hasParents)) {\n    return (\n      <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n        {Children.count(children) > 0 ? (\n          <StyledMenu\n            triggerMethod=\"hover\"\n            dynamicDomRendering={false} // As we parse the children we don't need dynamic rendering\n            disclosure={\n              <Button\n                sentiment=\"neutral\"\n                variant={hasActiveChildren ? 'filled' : 'ghost'}\n                size=\"small\"\n                icon={!categoryIcon ? 'dots-horizontal' : undefined}\n              >\n                {CategoryIconUsed ? (\n                  <Stack\n                    direction=\"row\"\n                    gap={1}\n                    alignItems=\"center\"\n                    justifyContent=\"center\"\n                  >\n                    <CategoryIconUsed\n                      variant={active ? 'primary' : categoryIconVariant}\n                    />\n                  </Stack>\n                ) : null}\n              </Button>\n            }\n            placement=\"right\"\n          >\n            {Children.map(children, child =>\n              isValidElement<ItemProps>(child)\n                ? cloneElement(child, {\n                    hasParents: true,\n                  })\n                : child,\n            )}\n          </StyledMenu>\n        ) : (\n          <Tooltip text={label} placement=\"right\" tabIndex={-1}>\n            <Button\n              sentiment=\"neutral\"\n              variant={active ? 'filled' : 'ghost'}\n              size=\"small\"\n            >\n              <Stack\n                direction=\"row\"\n                gap={1}\n                alignItems=\"center\"\n                justifyContent=\"center\"\n              >\n                {CategoryIconUsed ? (\n                  <CategoryIconUsed\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                ) : (\n                  <ConsoleCategoryIcon\n                    variant={active ? 'primary' : categoryIconVariant}\n                  />\n                )}\n              </Stack>\n            </Button>\n          </Tooltip>\n        )}\n      </MenuStack>\n    )\n  }\n\n  // This content is what is inside a menu item the navigation is collapsed\n  if (hasParents) {\n    return (\n      <StyledMenuItem\n        href={href}\n        borderless\n        active={active}\n        disabled={disabled}\n        sentiment={active ? 'primary' : 'neutral'}\n        isPinnable={shouldShowPinnedButton}\n        onClick={() => onToggle?.(!!active)}\n      >\n        <Stack\n          gap={1}\n          direction=\"row\"\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n          flex={1}\n          width=\"100%\"\n        >\n          <WrapText as=\"span\" variant=\"bodySmall\">\n            {label}\n          </WrapText>\n          <Stack direction=\"row\">\n            {badgeText ? (\n              <StyledBadge\n                sentiment={badgeSentiment}\n                size=\"small\"\n                prominence=\"strong\"\n                disabled={disabled}\n              >\n                {badgeText}\n              </StyledBadge>\n            ) : null}\n            {hasHrefAndNoChildren ? (\n              <AnimatedIcon\n                name=\"open-in-new\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                disabled={disabled}\n              />\n            ) : null}\n            {shouldShowPinnedButton ? (\n              <Tooltip\n                text={\n                  isItemPinned\n                    ? locales['navigation.unpin.tooltip']\n                    : pinTooltipLocale\n                }\n                placement=\"right\"\n              >\n                <RelativeDiv>\n                  <PinnedButton\n                    role=\"button\"\n                    size=\"xsmall\"\n                    aria-label={isItemPinned ? 'unpin' : 'pin'}\n                    variant=\"ghost\"\n                    sentiment={active ? 'primary' : 'neutral'}\n                    onClick={(event: MouseEvent<HTMLDivElement>) => {\n                      if (pinnedItems.length < pinLimit || isItemPinned) {\n                        event.preventDefault()\n                        event.stopPropagation() // This is to avoid click spread to the parent and change the routing\n\n                        let newValue: string[] | undefined\n                        if (isItemPinned) {\n                          newValue = unpinItem(id)\n                        } else {\n                          newValue = pinItem(id)\n                        }\n                        onClickPinUnpin?.({\n                          state: isItemPinned ? 'unpin' : 'pin',\n                          id,\n                          totalPinned: newValue,\n                        })\n                      }\n                    }}\n                    disabled={isItemPinned ? false : isPinDisabled}\n                  >\n                    <StyledIcon\n                      size=\"large\"\n                      name={isItemPinned ? 'unpin' : 'pin'}\n                      variant={isItemPinned ? 'filled' : 'outlined'}\n                      disabled={isItemPinned ? false : isPinDisabled}\n                      sentiment={active ? 'primary' : 'neutral'}\n                      active={active}\n                    />\n                  </PinnedButton>\n                </RelativeDiv>\n              </Tooltip>\n            ) : null}\n          </Stack>\n        </Stack>\n      </StyledMenuItem>\n    )\n  }\n\n  // This content is for when navigation is collapsed and we show an icon of link\n  if (!hasParents && href) {\n    return (\n      <Tooltip text={label} placement=\"right\">\n        <MenuStack gap={1} alignItems=\"start\" justifyContent=\"start\">\n          <Container\n            gap={1}\n            alignItems=\"center\"\n            justifyContent=\"center\"\n            href={href}\n            target=\"_blank\"\n          >\n            <AnimatedIcon\n              name=\"open-in-new\"\n              sentiment=\"neutral\"\n              prominence=\"weak\"\n            />\n          </Container>\n        </MenuStack>\n      </Tooltip>\n    )\n  }\n\n  return null\n}\n"]} */",
|
|
230
230
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
231
231
|
});
|
|
232
232
|
const Item = ({
|
|
@@ -381,19 +381,21 @@ const Item = ({
|
|
|
381
381
|
badgeText || hasPinnedFeatureAndNoChildren ? /* @__PURE__ */ jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [
|
|
382
382
|
badgeText ? /* @__PURE__ */ jsxRuntime.jsx(StyledBadge, { sentiment: badgeSentiment, size: "small", prominence: "strong", disabled, children: badgeText }) : null,
|
|
383
383
|
shouldShowPinnedButton ? /* @__PURE__ */ jsxRuntime.jsx(ui.Tooltip, { text: isItemPinned ? locales["navigation.unpin.tooltip"] : pinTooltipLocale, placement: "right", children: /* @__PURE__ */ jsxRuntime.jsx(RelativeDiv, { children: /* @__PURE__ */ jsxRuntime.jsx(PinnedButton, { role: "button", "aria-label": isItemPinned ? "unpin" : "pin", size: "xsmall", variant: "ghost", sentiment: active ? "primary" : "neutral", onClick: (event) => {
|
|
384
|
-
|
|
385
|
-
|
|
386
|
-
|
|
387
|
-
|
|
388
|
-
|
|
389
|
-
|
|
390
|
-
|
|
384
|
+
if (pinnedItems.length < pinLimit || isItemPinned) {
|
|
385
|
+
event.preventDefault();
|
|
386
|
+
event.stopPropagation();
|
|
387
|
+
let newValue;
|
|
388
|
+
if (isItemPinned) {
|
|
389
|
+
newValue = unpinItem(id);
|
|
390
|
+
} else {
|
|
391
|
+
newValue = pinItem(id);
|
|
392
|
+
}
|
|
393
|
+
onClickPinUnpin?.({
|
|
394
|
+
state: isItemPinned ? "unpin" : "pin",
|
|
395
|
+
id,
|
|
396
|
+
totalPinned: newValue
|
|
397
|
+
});
|
|
391
398
|
}
|
|
392
|
-
onClickPinUnpin?.({
|
|
393
|
-
state: isItemPinned ? "unpin" : "pin",
|
|
394
|
-
id,
|
|
395
|
-
totalPinned: newValue
|
|
396
|
-
});
|
|
397
399
|
}, disabled: isItemPinned ? false : isPinDisabled, children: /* @__PURE__ */ jsxRuntime.jsx(StyledIcon, { size: "large", name: isItemPinned ? "unpin" : "pin", variant: isItemPinned ? "filled" : "outlined", disabled: isItemPinned ? false : isPinDisabled, sentiment: active ? "primary" : "neutral", active }) }) }) }) : null
|
|
398
400
|
] }) : null,
|
|
399
401
|
hasHrefAndNoChildren ? /* @__PURE__ */ jsxRuntime.jsx(AnimatedIcon, { name: "open-in-new", sentiment: "neutral", prominence: "default", disabled }) : null,
|
|
@@ -424,19 +426,21 @@ const Item = ({
|
|
|
424
426
|
badgeText ? /* @__PURE__ */ jsxRuntime.jsx(StyledBadge, { sentiment: badgeSentiment, size: "small", prominence: "strong", disabled, children: badgeText }) : null,
|
|
425
427
|
hasHrefAndNoChildren ? /* @__PURE__ */ jsxRuntime.jsx(AnimatedIcon, { name: "open-in-new", sentiment: "neutral", prominence: "weak", disabled }) : null,
|
|
426
428
|
shouldShowPinnedButton ? /* @__PURE__ */ jsxRuntime.jsx(ui.Tooltip, { text: isItemPinned ? locales["navigation.unpin.tooltip"] : pinTooltipLocale, placement: "right", children: /* @__PURE__ */ jsxRuntime.jsx(RelativeDiv, { children: /* @__PURE__ */ jsxRuntime.jsx(PinnedButton, { role: "button", size: "xsmall", "aria-label": isItemPinned ? "unpin" : "pin", variant: "ghost", sentiment: active ? "primary" : "neutral", onClick: (event) => {
|
|
427
|
-
|
|
428
|
-
|
|
429
|
-
|
|
430
|
-
|
|
431
|
-
|
|
432
|
-
|
|
433
|
-
|
|
429
|
+
if (pinnedItems.length < pinLimit || isItemPinned) {
|
|
430
|
+
event.preventDefault();
|
|
431
|
+
event.stopPropagation();
|
|
432
|
+
let newValue;
|
|
433
|
+
if (isItemPinned) {
|
|
434
|
+
newValue = unpinItem(id);
|
|
435
|
+
} else {
|
|
436
|
+
newValue = pinItem(id);
|
|
437
|
+
}
|
|
438
|
+
onClickPinUnpin?.({
|
|
439
|
+
state: isItemPinned ? "unpin" : "pin",
|
|
440
|
+
id,
|
|
441
|
+
totalPinned: newValue
|
|
442
|
+
});
|
|
434
443
|
}
|
|
435
|
-
onClickPinUnpin?.({
|
|
436
|
-
state: isItemPinned ? "unpin" : "pin",
|
|
437
|
-
id,
|
|
438
|
-
totalPinned: newValue
|
|
439
|
-
});
|
|
440
444
|
}, disabled: isItemPinned ? false : isPinDisabled, children: /* @__PURE__ */ jsxRuntime.jsx(StyledIcon, { size: "large", name: isItemPinned ? "unpin" : "pin", variant: isItemPinned ? "filled" : "outlined", disabled: isItemPinned ? false : isPinDisabled, sentiment: active ? "primary" : "neutral", active }) }) }) }) : null
|
|
441
445
|
] })
|
|
442
446
|
] }) });
|