td-stylekit 30.4.0 → 30.4.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CHANGELOG.md CHANGED
@@ -1,3 +1,17 @@
1
+ ## [30.4.2](https://github.com/treasure-data/td-stylekit/compare/v30.4.1...v30.4.2) (2025-01-27)
2
+
3
+
4
+ ### Bug Fixes
5
+
6
+ * **CON-17736:** Do not allow nested Icons in Buttons to swallow up events ([#1622](https://github.com/treasure-data/td-stylekit/issues/1622)) ([c30cfa0](https://github.com/treasure-data/td-stylekit/commit/c30cfa04c205fa80b890290e99ab89723c77cc3e))
7
+
8
+ ## [30.4.1](https://github.com/treasure-data/td-stylekit/compare/v30.4.0...v30.4.1) (2024-11-28)
9
+
10
+
11
+ ### Bug Fixes
12
+
13
+ * **CON-17518:** Remove extra line when we select new element in Tagger ([#1616](https://github.com/treasure-data/td-stylekit/issues/1616)) ([7292850](https://github.com/treasure-data/td-stylekit/commit/72928505cd2e2fe0a8952b92c03fc6bfff7a5780))
14
+
1
15
  # [30.4.0](https://github.com/treasure-data/td-stylekit/compare/v30.3.0...v30.4.0) (2024-11-26)
2
16
 
3
17
 
@@ -167,7 +167,12 @@ function (_ref6) {
167
167
  height: 'auto',
168
168
  padding: 0,
169
169
  minHeight: small ? (0, _utils.multiply)(theme.space[2], 2.5) : (0, _utils.multiply)(theme.space[3], 3),
170
- minWidth: small ? (0, _utils.multiply)(theme.space[2], 2.5) : (0, _utils.multiply)(theme.space[3], 3)
170
+ minWidth: small ? (0, _utils.multiply)(theme.space[2], 2.5) : (0, _utils.multiply)(theme.space[3], 3),
171
+ 'svg:first-of-type': {
172
+ // prevents nested icons from swallowing click events
173
+ // https://css-tricks.com/slightly-careful-sub-elements-clickable-things/
174
+ pointerEvents: 'none'
175
+ }
171
176
  };
172
177
  },
173
178
  // circle
@@ -394,7 +399,7 @@ var StyledButton = /*#__PURE__*/(0, _base["default"])(UnstyledButton, process.en
394
399
  } : {
395
400
  target: "e1tp8vho0",
396
401
  label: "StyledButton"
397
- })(styles, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/Button/Button.tsx"],"names":[],"mappings":"AA4fqB","file":"../../../src/Button/Button.tsx","sourcesContent":["import {\n  PropsWithChildren,\n  RefObject,\n  ButtonHTMLAttributes,\n  Component,\n  forwardRef,\n  createRef,\n  ReactNode,\n  ComponentType,\n  SyntheticEvent,\n  KeyboardEvent\n} from 'react'\nimport color from 'color'\nimport styled from '@emotion/styled'\nimport omit from 'lodash.omit'\nimport { multiply } from '../utils'\nimport { getOverrides, Overridable } from '../ThemeProvider'\nimport type { Theme } from '../ThemeProvider'\nimport { Spinner } from './elements'\n\nconst styles: Array<(props: any) => any> = [\n  // reset\n  ({\n    disabled,\n    theme\n  }: {\n    ignoreFocus: boolean\n    disabled: boolean\n    loading: boolean\n    theme: Theme\n  }) => ({\n    alignItems: 'center',\n    backgroundColor: 'transparent',\n    border: '1px solid transparent',\n    cursor: disabled ? 'default' : 'pointer',\n    display: 'inline-flex',\n    flexShrink: 0,\n    fontFamily: theme.fontFamily.body,\n    justifyContent: 'center',\n    margin: 0,\n    padding: 0,\n    position: 'relative',\n    textAlign: 'center',\n    textDecoration: 'none',\n    textRendering: 'optimizeLegibility',\n    touchAction: 'manipulation',\n    userSelect: 'none',\n    verticalAlign: 'middle',\n    whiteSpace: 'nowrap'\n  }),\n  // default outline\n  ({\n    theme,\n    plain,\n    icon,\n    disabled,\n    loading\n  }: {\n    theme: Theme\n    plain: boolean\n    icon: boolean\n    disabled: boolean\n    loading: boolean\n  }) =>\n    !plain &&\n    !icon && {\n      backgroundColor: theme.palette.neutral[0],\n      border: disabled\n        ? `1px solid ${color(theme.palette.primary[4]).alpha(0.3).string()}`\n        : `1px solid ${theme.palette.primary[4]}`,\n      borderRadius: theme.radius[4],\n      color: disabled\n        ? color(theme.palette.primary[4]).alpha(0.3).string()\n        : theme.palette.primary[4],\n      fontSize: theme.fontSize[0],\n      fontWeight: 600,\n      height: theme.space[8],\n      lineHeight: theme.lineHeight[4],\n      overflow: 'hidden',\n      padding: `0 ${theme.space[4]}`,\n      textOverflow: 'ellipsis',\n      outline: 'none',\n      textTransform: 'unset',\n      transition: 'transform 0.15s ease-in-out',\n      ':hover': !disabled &&\n        !loading && {\n          boxShadow: '0px 2px 7px rgba(0, 0, 0, 0.2)',\n          transform: 'translateY(-1px)'\n        },\n      ':active': !disabled &&\n        !loading && {\n          backgroundColor: theme.palette.primary[0],\n          boxShadow: `inset 0px 2px 4px ${color(theme.palette.primary[4])\n            .darken(0.1)\n            .string()}`\n        }\n    },\n\n  // primary\n  ({\n    theme,\n    primary,\n    disabled,\n    loading\n  }: {\n    theme: Theme\n    primary: boolean\n    disabled: boolean\n    loading: boolean\n  }) =>\n    primary && {\n      backgroundColor: disabled\n        ? color(theme.palette.primary[4]).alpha(0.3).string()\n        : theme.palette.primary[4],\n      color: theme.palette.neutral[0],\n      borderColor: 'transparent',\n      ':hover': !disabled &&\n        !loading && {\n          backgroundColor: color(theme.palette.primary[4])\n            .lighten(0.4)\n            .string(),\n          boxShadow: `0 2px 7px ${color(theme.palette.primary[4])\n            .lighten(0.4)\n            .alpha(0.35)\n            .string()}`\n        },\n      ':active': !disabled &&\n        !loading && {\n          backgroundColor: theme.palette.primary[5],\n          boxShadow: `inset 0px 2px 4px ${color(theme.palette.primary[5])\n            .darken(0.425)\n            .string()}`,\n          color: theme.palette.neutral[0]\n        }\n    },\n\n  // borderless\n  ({\n    theme,\n    borderless,\n    disabled,\n    loading\n  }: {\n    theme: Theme\n    borderless: boolean\n    disabled: boolean\n    loading: boolean\n  }) =>\n    borderless && {\n      backgroundColor: 'transparent',\n      border: 'none',\n      color: disabled\n        ? color(theme.palette.primary[4]).alpha(0.3).string()\n        : theme.palette.primary[4],\n      borderColor: 'none',\n      ':hover': !disabled &&\n        !loading && {\n          boxShadow: 'none',\n          textDecoration: 'underline',\n          transform: 'none'\n        },\n      ':active': !disabled &&\n        !loading && {\n          background: 'transparent',\n          boxShadow: 'none'\n        }\n    },\n  // borderless Alt\n  ({\n    theme,\n    borderlessAlt,\n    disabled,\n    loading\n  }: {\n    theme: Theme\n    borderlessAlt: boolean\n    disabled: boolean\n    loading: boolean\n  }) =>\n    borderlessAlt && {\n      background: 'transparent',\n      color: disabled\n        ? color(theme.palette.neutral[0]).alpha(0.5).string()\n        : theme.palette.neutral[0],\n      borderColor: 'none',\n      border: 'none',\n      ':hover': !disabled &&\n        !loading && {\n          boxShadow: `none`,\n          textDecoration: 'underline',\n          transform: 'none'\n        },\n      ':active': !disabled &&\n        !loading && {\n          background: 'transparent',\n          textDecoration: 'underline'\n        }\n    },\n  // icon\n  ({ theme, icon, small }: { theme: Theme; icon: boolean; small: boolean }) =>\n    icon && {\n      borderRadius: 0,\n      height: 'auto',\n      padding: 0,\n      minHeight: small\n        ? multiply(theme.space[2], 2.5)\n        : multiply(theme.space[3], 3),\n      minWidth: small\n        ? multiply(theme.space[2], 2.5)\n        : multiply(theme.space[3], 3)\n    },\n  // circle\n  ({ theme, circle }: { theme: Theme; circle: boolean }) =>\n    circle && {\n      borderRadius: '50%',\n      height: multiply(theme.space[3], 2),\n      lineHeight: theme.lineHeight[4],\n      padding: 0,\n      overflow: 'hidden',\n      width: multiply(theme.space[3], 2),\n      ':hover': {\n        transform: 'none'\n      }\n    },\n  // danger\n  ({\n    theme,\n    danger,\n    disabled,\n    loading\n  }: {\n    theme: Theme\n    danger: boolean\n    disabled: boolean\n    loading: boolean\n  }) =>\n    danger && {\n      backgroundColor: disabled\n        ? color(theme.palette.error[1]).alpha(0.3).string()\n        : theme.palette.error[1],\n      color: theme.palette.neutral[0],\n      border: 'none',\n      ':hover': !disabled &&\n        !loading && {\n          backgroundColor: color(theme.palette.error[1]).lighten(0.2).string(),\n          boxShadow: `0 2px 7px ${color(theme.palette.error[1])\n            .lighten(0.2)\n            .string()}`\n        },\n      ':active': !disabled &&\n        !loading && {\n          backgroundColor: color(theme.palette.error[1]).darken(0.1).string(),\n          borderColor: 'transparent',\n          color: theme.palette.neutral[0],\n          boxShadow: `inset 0px 2px 4px ${color(theme.palette.error[1])\n            .darken(0.3)\n            .string()}`\n        }\n    },\n  // dangerAlt\n  ({\n    theme,\n    dangerAlt,\n    disabled,\n    loading\n  }: {\n    theme: Theme\n    dangerAlt: boolean\n    disabled: boolean\n    loading: boolean\n  }) =>\n    dangerAlt && {\n      backgroundColor: disabled\n        ? color(theme.palette.neutral[0]).alpha(0.5).string()\n        : theme.palette.neutral[0],\n      color: theme.palette.error[1],\n      border: 'none',\n      ':hover': !disabled &&\n        !loading && { boxShadow: `0 2px 7px rgba(0, 0, 0, 0.5)` },\n      ':active': !disabled &&\n        !loading && {\n          backgroundColor: theme.palette.neutral[0],\n          borderColor: 'transparent',\n          color: theme.palette.error[1],\n          boxShadow: `inset 0px 2px 4px ${color(theme.palette.neutral[0])\n            .darken(0.3)\n            .string()}`\n        }\n    },\n  // loading\n  ({ theme, loading }: { theme: Theme; loading: boolean }) =>\n    loading && {\n      backgroundColor: theme.palette.neutral[3],\n      border: 'none',\n      color: theme.palette.neutral[9],\n      cursor: 'default'\n    },\n  // noFocus\n  ({ ignoreFocus }: { ignoreFocus: boolean }) =>\n    ignoreFocus && {\n      outline: 'none',\n      ':focus': {\n        outline: 'none'\n      }\n    },\n  // coloredFocus\n  ({ theme, colorFocus }: { theme: Theme; colorFocus: boolean }) =>\n    colorFocus && {\n      outline: 'none',\n      ':focus': {\n        outline: 'none',\n        color: theme.palette.primary[4]\n      }\n    },\n  getOverrides(Overridable.Button.Root)\n]\n\nexport type UnstyledButtonProps = {\n  /** Focus the button when it's rendered */\n  autofocus?: boolean\n  /** Unfocus the button when it's clicked */\n  blurOnClick?: boolean\n  /** Text Button */\n  borderless?: boolean\n  /** Text Button on non-white Background */\n  borderlessAlt?: boolean\n  children: ReactNode\n  /** Enable circle style */\n  circle?: boolean\n  className?: string\n  /** Removes outline and changes color to Primary on focus */\n  colorFocus?: boolean\n  /** Stateful component or html tag to render as (defaults to `button`) */\n  component: ComponentType<PropsWithChildren<any>> | string\n  /** Enable danger style */\n  danger?: boolean\n  /** Instrumentation */\n  'data-instrumentation'?: string\n  /** Enable dangerAlt style --- white background */\n  dangerAlt?: boolean\n  /** Enable disabled style */\n  disabled?: boolean\n  /** Title to display when button is disabled */\n  disabledTitle?: string\n  getRef?: RefObject<HTMLElement>\n  form?: string\n  href?: string\n  id?: string\n  /** Enable styles suitible for wrapping icons */\n  icon?: boolean\n  /** Prevent button from being focused */\n  ignoreFocus?: boolean\n  loading?: boolean\n  onClick?: (event: SyntheticEvent<HTMLButtonElement>) => void\n  onKeyPress?: (event: KeyboardEvent<HTMLButtonElement>) => void\n  onMouseDown?: (event: KeyboardEvent<HTMLButtonElement>) => void\n  /** Disable default button styles */\n  plain?: boolean\n  /** Fill button with solid color */\n  primary?: boolean\n  rel?: string\n  /** Only used with `icon`. */\n  small?: boolean\n  target?: string\n  theme?: Theme\n  /** Title to display when button is enabled */\n  title?: string\n  /** HTML button type attribute */\n  type?: ButtonHTMLAttributes<HTMLButtonElement>['type']\n  hasError?: boolean\n  /** Indicates whether the button should not have a special visual appearance when mouse hovers above it. */\n  hoverless?: boolean\n}\n\nexport class UnstyledButton extends Component<UnstyledButtonProps> {\n  static defaultProps = {\n    borderless: false,\n    borderlessAlt: false,\n    blurOnClick: false,\n    circle: false,\n    component: 'button',\n    danger: false,\n    dangerAlt: false,\n    'data-instrumentation': 'button',\n    disabled: false,\n    loading: false,\n    small: false,\n    primary: false,\n    hoverless: false\n  }\n\n  /**\n   * https://mathiasbynens.github.io/rel-noopener/\n   * Using target=_blank is insecure, so we add\n   * rel=noopener to reduce the attack surface\n   */\n  static getRel({ target, rel }: { target?: string; rel?: string }) {\n    const relProp: string[] = []\n    if (rel) {\n      relProp.push(rel)\n    }\n    if (target === '_blank') {\n      relProp.push('noopener')\n    }\n    if (relProp.length) {\n      return relProp.join(' ')\n    }\n    return undefined\n  }\n\n  constructor(props: UnstyledButtonProps) {\n    super(props)\n    this.buttonRef = props.getRef || createRef()\n  }\n\n  buttonRef: RefObject<HTMLElement>\n\n  _onKeyPress = (event: KeyboardEvent<HTMLButtonElement>) => {\n    const { component, disabled, loading, onClick, onKeyPress } = this.props\n    if (!disabled && !loading) {\n      if (\n        onClick &&\n        ['Enter', ' '].includes(event.key) &&\n        component !== 'button'\n      ) {\n        event.preventDefault()\n        onClick(event)\n        return\n      }\n      if (onKeyPress) {\n        onKeyPress(event)\n      }\n    }\n  }\n\n  _onClick = (event: SyntheticEvent<HTMLButtonElement>) => {\n    const { disabled, loading, onClick, blurOnClick } = this.props\n    if (!disabled && !loading && onClick) {\n      onClick(event)\n    }\n    if (blurOnClick) {\n      if (this.buttonRef && this.buttonRef.current) {\n        this.buttonRef.current.blur()\n      }\n    }\n  }\n\n  componentDidMount() {\n    if (this.props.autofocus) {\n      this.buttonRef && this.buttonRef.current && this.buttonRef.current.focus()\n    }\n  }\n\n  render() {\n    const {\n      children,\n      component: ButtonComponent,\n      'data-instrumentation': dataInstrumentation,\n      disabled,\n      disabledTitle,\n      loading,\n      rel,\n      target,\n      title,\n      ...props\n    } = this.props\n\n    const buttonProps = omit(props, [\n      'autofocus',\n      'borderless',\n      'borderlessAlt',\n      'blurOnClick',\n      'circle',\n      'danger',\n      'dangerAlt',\n      'ignoreFocus',\n      'colorFocus',\n      'hoverless',\n      'icon',\n      'loading',\n      'plain',\n      'primary',\n      'small',\n      'square',\n      'getRef'\n    ])\n\n    return (\n      <ButtonComponent\n        tabIndex={disabled ? -1 : 0}\n        role=\"button\"\n        {...buttonProps}\n        data-instrumentation={dataInstrumentation}\n        disabled={loading || disabled}\n        onClick={this._onClick}\n        onKeyPress={this._onKeyPress}\n        ref={this.buttonRef}\n        rel={UnstyledButton.getRel({ rel, target })}\n        title={disabled ? disabledTitle : title}\n        target={target}\n      >\n        {children}\n        {loading && <Spinner />}\n      </ButtonComponent>\n    )\n  }\n}\n\nconst StyledButton = styled(UnstyledButton)(styles)\n// eslint-disable-next-line react/display-name\nexport const Button = forwardRef<UnstyledButton, Partial<UnstyledButtonProps>>(\n  (props, ref) => <StyledButton getRef={ref} {...props} />\n)\n\nexport default Button\n"]} */");
402
+ })(styles, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/Button/Button.tsx"],"names":[],"mappings":"AAigBqB","file":"../../../src/Button/Button.tsx","sourcesContent":["import {\n  PropsWithChildren,\n  RefObject,\n  ButtonHTMLAttributes,\n  Component,\n  forwardRef,\n  createRef,\n  ReactNode,\n  ComponentType,\n  SyntheticEvent,\n  KeyboardEvent\n} from 'react'\nimport color from 'color'\nimport styled from '@emotion/styled'\nimport omit from 'lodash.omit'\nimport { multiply } from '../utils'\nimport { getOverrides, Overridable } from '../ThemeProvider'\nimport type { Theme } from '../ThemeProvider'\nimport { Spinner } from './elements'\n\nconst styles: Array<(props: any) => any> = [\n  // reset\n  ({\n    disabled,\n    theme\n  }: {\n    ignoreFocus: boolean\n    disabled: boolean\n    loading: boolean\n    theme: Theme\n  }) => ({\n    alignItems: 'center',\n    backgroundColor: 'transparent',\n    border: '1px solid transparent',\n    cursor: disabled ? 'default' : 'pointer',\n    display: 'inline-flex',\n    flexShrink: 0,\n    fontFamily: theme.fontFamily.body,\n    justifyContent: 'center',\n    margin: 0,\n    padding: 0,\n    position: 'relative',\n    textAlign: 'center',\n    textDecoration: 'none',\n    textRendering: 'optimizeLegibility',\n    touchAction: 'manipulation',\n    userSelect: 'none',\n    verticalAlign: 'middle',\n    whiteSpace: 'nowrap'\n  }),\n  // default outline\n  ({\n    theme,\n    plain,\n    icon,\n    disabled,\n    loading\n  }: {\n    theme: Theme\n    plain: boolean\n    icon: boolean\n    disabled: boolean\n    loading: boolean\n  }) =>\n    !plain &&\n    !icon && {\n      backgroundColor: theme.palette.neutral[0],\n      border: disabled\n        ? `1px solid ${color(theme.palette.primary[4]).alpha(0.3).string()}`\n        : `1px solid ${theme.palette.primary[4]}`,\n      borderRadius: theme.radius[4],\n      color: disabled\n        ? color(theme.palette.primary[4]).alpha(0.3).string()\n        : theme.palette.primary[4],\n      fontSize: theme.fontSize[0],\n      fontWeight: 600,\n      height: theme.space[8],\n      lineHeight: theme.lineHeight[4],\n      overflow: 'hidden',\n      padding: `0 ${theme.space[4]}`,\n      textOverflow: 'ellipsis',\n      outline: 'none',\n      textTransform: 'unset',\n      transition: 'transform 0.15s ease-in-out',\n      ':hover': !disabled &&\n        !loading && {\n          boxShadow: '0px 2px 7px rgba(0, 0, 0, 0.2)',\n          transform: 'translateY(-1px)'\n        },\n      ':active': !disabled &&\n        !loading && {\n          backgroundColor: theme.palette.primary[0],\n          boxShadow: `inset 0px 2px 4px ${color(theme.palette.primary[4])\n            .darken(0.1)\n            .string()}`\n        }\n    },\n\n  // primary\n  ({\n    theme,\n    primary,\n    disabled,\n    loading\n  }: {\n    theme: Theme\n    primary: boolean\n    disabled: boolean\n    loading: boolean\n  }) =>\n    primary && {\n      backgroundColor: disabled\n        ? color(theme.palette.primary[4]).alpha(0.3).string()\n        : theme.palette.primary[4],\n      color: theme.palette.neutral[0],\n      borderColor: 'transparent',\n      ':hover': !disabled &&\n        !loading && {\n          backgroundColor: color(theme.palette.primary[4])\n            .lighten(0.4)\n            .string(),\n          boxShadow: `0 2px 7px ${color(theme.palette.primary[4])\n            .lighten(0.4)\n            .alpha(0.35)\n            .string()}`\n        },\n      ':active': !disabled &&\n        !loading && {\n          backgroundColor: theme.palette.primary[5],\n          boxShadow: `inset 0px 2px 4px ${color(theme.palette.primary[5])\n            .darken(0.425)\n            .string()}`,\n          color: theme.palette.neutral[0]\n        }\n    },\n\n  // borderless\n  ({\n    theme,\n    borderless,\n    disabled,\n    loading\n  }: {\n    theme: Theme\n    borderless: boolean\n    disabled: boolean\n    loading: boolean\n  }) =>\n    borderless && {\n      backgroundColor: 'transparent',\n      border: 'none',\n      color: disabled\n        ? color(theme.palette.primary[4]).alpha(0.3).string()\n        : theme.palette.primary[4],\n      borderColor: 'none',\n      ':hover': !disabled &&\n        !loading && {\n          boxShadow: 'none',\n          textDecoration: 'underline',\n          transform: 'none'\n        },\n      ':active': !disabled &&\n        !loading && {\n          background: 'transparent',\n          boxShadow: 'none'\n        }\n    },\n  // borderless Alt\n  ({\n    theme,\n    borderlessAlt,\n    disabled,\n    loading\n  }: {\n    theme: Theme\n    borderlessAlt: boolean\n    disabled: boolean\n    loading: boolean\n  }) =>\n    borderlessAlt && {\n      background: 'transparent',\n      color: disabled\n        ? color(theme.palette.neutral[0]).alpha(0.5).string()\n        : theme.palette.neutral[0],\n      borderColor: 'none',\n      border: 'none',\n      ':hover': !disabled &&\n        !loading && {\n          boxShadow: `none`,\n          textDecoration: 'underline',\n          transform: 'none'\n        },\n      ':active': !disabled &&\n        !loading && {\n          background: 'transparent',\n          textDecoration: 'underline'\n        }\n    },\n  // icon\n  ({ theme, icon, small }: { theme: Theme; icon: boolean; small: boolean }) =>\n    icon && {\n      borderRadius: 0,\n      height: 'auto',\n      padding: 0,\n      minHeight: small\n        ? multiply(theme.space[2], 2.5)\n        : multiply(theme.space[3], 3),\n      minWidth: small\n        ? multiply(theme.space[2], 2.5)\n        : multiply(theme.space[3], 3),\n      'svg:first-of-type': {\n        // prevents nested icons from swallowing click events\n        // https://css-tricks.com/slightly-careful-sub-elements-clickable-things/\n        pointerEvents: 'none'\n      }\n    },\n  // circle\n  ({ theme, circle }: { theme: Theme; circle: boolean }) =>\n    circle && {\n      borderRadius: '50%',\n      height: multiply(theme.space[3], 2),\n      lineHeight: theme.lineHeight[4],\n      padding: 0,\n      overflow: 'hidden',\n      width: multiply(theme.space[3], 2),\n      ':hover': {\n        transform: 'none'\n      }\n    },\n  // danger\n  ({\n    theme,\n    danger,\n    disabled,\n    loading\n  }: {\n    theme: Theme\n    danger: boolean\n    disabled: boolean\n    loading: boolean\n  }) =>\n    danger && {\n      backgroundColor: disabled\n        ? color(theme.palette.error[1]).alpha(0.3).string()\n        : theme.palette.error[1],\n      color: theme.palette.neutral[0],\n      border: 'none',\n      ':hover': !disabled &&\n        !loading && {\n          backgroundColor: color(theme.palette.error[1]).lighten(0.2).string(),\n          boxShadow: `0 2px 7px ${color(theme.palette.error[1])\n            .lighten(0.2)\n            .string()}`\n        },\n      ':active': !disabled &&\n        !loading && {\n          backgroundColor: color(theme.palette.error[1]).darken(0.1).string(),\n          borderColor: 'transparent',\n          color: theme.palette.neutral[0],\n          boxShadow: `inset 0px 2px 4px ${color(theme.palette.error[1])\n            .darken(0.3)\n            .string()}`\n        }\n    },\n  // dangerAlt\n  ({\n    theme,\n    dangerAlt,\n    disabled,\n    loading\n  }: {\n    theme: Theme\n    dangerAlt: boolean\n    disabled: boolean\n    loading: boolean\n  }) =>\n    dangerAlt && {\n      backgroundColor: disabled\n        ? color(theme.palette.neutral[0]).alpha(0.5).string()\n        : theme.palette.neutral[0],\n      color: theme.palette.error[1],\n      border: 'none',\n      ':hover': !disabled &&\n        !loading && { boxShadow: `0 2px 7px rgba(0, 0, 0, 0.5)` },\n      ':active': !disabled &&\n        !loading && {\n          backgroundColor: theme.palette.neutral[0],\n          borderColor: 'transparent',\n          color: theme.palette.error[1],\n          boxShadow: `inset 0px 2px 4px ${color(theme.palette.neutral[0])\n            .darken(0.3)\n            .string()}`\n        }\n    },\n  // loading\n  ({ theme, loading }: { theme: Theme; loading: boolean }) =>\n    loading && {\n      backgroundColor: theme.palette.neutral[3],\n      border: 'none',\n      color: theme.palette.neutral[9],\n      cursor: 'default'\n    },\n  // noFocus\n  ({ ignoreFocus }: { ignoreFocus: boolean }) =>\n    ignoreFocus && {\n      outline: 'none',\n      ':focus': {\n        outline: 'none'\n      }\n    },\n  // coloredFocus\n  ({ theme, colorFocus }: { theme: Theme; colorFocus: boolean }) =>\n    colorFocus && {\n      outline: 'none',\n      ':focus': {\n        outline: 'none',\n        color: theme.palette.primary[4]\n      }\n    },\n  getOverrides(Overridable.Button.Root)\n]\n\nexport type UnstyledButtonProps = {\n  /** Focus the button when it's rendered */\n  autofocus?: boolean\n  /** Unfocus the button when it's clicked */\n  blurOnClick?: boolean\n  /** Text Button */\n  borderless?: boolean\n  /** Text Button on non-white Background */\n  borderlessAlt?: boolean\n  children: ReactNode\n  /** Enable circle style */\n  circle?: boolean\n  className?: string\n  /** Removes outline and changes color to Primary on focus */\n  colorFocus?: boolean\n  /** Stateful component or html tag to render as (defaults to `button`) */\n  component: ComponentType<PropsWithChildren<any>> | string\n  /** Enable danger style */\n  danger?: boolean\n  /** Instrumentation */\n  'data-instrumentation'?: string\n  /** Enable dangerAlt style --- white background */\n  dangerAlt?: boolean\n  /** Enable disabled style */\n  disabled?: boolean\n  /** Title to display when button is disabled */\n  disabledTitle?: string\n  getRef?: RefObject<HTMLElement>\n  form?: string\n  href?: string\n  id?: string\n  /** Enable styles suitible for wrapping icons */\n  icon?: boolean\n  /** Prevent button from being focused */\n  ignoreFocus?: boolean\n  loading?: boolean\n  onClick?: (event: SyntheticEvent<HTMLButtonElement>) => void\n  onKeyPress?: (event: KeyboardEvent<HTMLButtonElement>) => void\n  onMouseDown?: (event: KeyboardEvent<HTMLButtonElement>) => void\n  /** Disable default button styles */\n  plain?: boolean\n  /** Fill button with solid color */\n  primary?: boolean\n  rel?: string\n  /** Only used with `icon`. */\n  small?: boolean\n  target?: string\n  theme?: Theme\n  /** Title to display when button is enabled */\n  title?: string\n  /** HTML button type attribute */\n  type?: ButtonHTMLAttributes<HTMLButtonElement>['type']\n  hasError?: boolean\n  /** Indicates whether the button should not have a special visual appearance when mouse hovers above it. */\n  hoverless?: boolean\n}\n\nexport class UnstyledButton extends Component<UnstyledButtonProps> {\n  static defaultProps = {\n    borderless: false,\n    borderlessAlt: false,\n    blurOnClick: false,\n    circle: false,\n    component: 'button',\n    danger: false,\n    dangerAlt: false,\n    'data-instrumentation': 'button',\n    disabled: false,\n    loading: false,\n    small: false,\n    primary: false,\n    hoverless: false\n  }\n\n  /**\n   * https://mathiasbynens.github.io/rel-noopener/\n   * Using target=_blank is insecure, so we add\n   * rel=noopener to reduce the attack surface\n   */\n  static getRel({ target, rel }: { target?: string; rel?: string }) {\n    const relProp: string[] = []\n    if (rel) {\n      relProp.push(rel)\n    }\n    if (target === '_blank') {\n      relProp.push('noopener')\n    }\n    if (relProp.length) {\n      return relProp.join(' ')\n    }\n    return undefined\n  }\n\n  constructor(props: UnstyledButtonProps) {\n    super(props)\n    this.buttonRef = props.getRef || createRef()\n  }\n\n  buttonRef: RefObject<HTMLElement>\n\n  _onKeyPress = (event: KeyboardEvent<HTMLButtonElement>) => {\n    const { component, disabled, loading, onClick, onKeyPress } = this.props\n    if (!disabled && !loading) {\n      if (\n        onClick &&\n        ['Enter', ' '].includes(event.key) &&\n        component !== 'button'\n      ) {\n        event.preventDefault()\n        onClick(event)\n        return\n      }\n      if (onKeyPress) {\n        onKeyPress(event)\n      }\n    }\n  }\n\n  _onClick = (event: SyntheticEvent<HTMLButtonElement>) => {\n    const { disabled, loading, onClick, blurOnClick } = this.props\n    if (!disabled && !loading && onClick) {\n      onClick(event)\n    }\n    if (blurOnClick) {\n      if (this.buttonRef && this.buttonRef.current) {\n        this.buttonRef.current.blur()\n      }\n    }\n  }\n\n  componentDidMount() {\n    if (this.props.autofocus) {\n      this.buttonRef && this.buttonRef.current && this.buttonRef.current.focus()\n    }\n  }\n\n  render() {\n    const {\n      children,\n      component: ButtonComponent,\n      'data-instrumentation': dataInstrumentation,\n      disabled,\n      disabledTitle,\n      loading,\n      rel,\n      target,\n      title,\n      ...props\n    } = this.props\n\n    const buttonProps = omit(props, [\n      'autofocus',\n      'borderless',\n      'borderlessAlt',\n      'blurOnClick',\n      'circle',\n      'danger',\n      'dangerAlt',\n      'ignoreFocus',\n      'colorFocus',\n      'hoverless',\n      'icon',\n      'loading',\n      'plain',\n      'primary',\n      'small',\n      'square',\n      'getRef'\n    ])\n\n    return (\n      <ButtonComponent\n        tabIndex={disabled ? -1 : 0}\n        role=\"button\"\n        {...buttonProps}\n        data-instrumentation={dataInstrumentation}\n        disabled={loading || disabled}\n        onClick={this._onClick}\n        onKeyPress={this._onKeyPress}\n        ref={this.buttonRef}\n        rel={UnstyledButton.getRel({ rel, target })}\n        title={disabled ? disabledTitle : title}\n        target={target}\n      >\n        {children}\n        {loading && <Spinner />}\n      </ButtonComponent>\n    )\n  }\n}\n\nconst StyledButton = styled(UnstyledButton)(styles)\n// eslint-disable-next-line react/display-name\nexport const Button = forwardRef<UnstyledButton, Partial<UnstyledButtonProps>>(\n  (props, ref) => <StyledButton getRef={ref} {...props} />\n)\n\nexport default Button\n"]} */");
398
403
  // eslint-disable-next-line react/display-name
399
404
  var Button = exports.Button = /*#__PURE__*/(0, _react.forwardRef)(function (props, ref) {
400
405
  return (0, _jsxRuntime.jsx)(StyledButton, _objectSpread({
@@ -64,7 +64,7 @@ var DefaultOptionFormatterContainer = /*#__PURE__*/(0, _base["default"])('div',
64
64
  } : {
65
65
  name: "l8l8b8",
66
66
  styles: "white-space:nowrap;overflow:hidden;text-overflow:ellipsis",
67
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/Tagger/components/PopupSelector.tsx"],"names":[],"mappings":"AAgDwC","file":"../../../../src/Tagger/components/PopupSelector.tsx","sourcesContent":["/* eslint-disable td/instrumentation */\n// this level of fine grained instrumentation is not needed\nimport {\n  ReactNode,\n  RefObject,\n  CSSProperties,\n  ComponentProps,\n  KeyboardEvent,\n  createRef,\n  PureComponent\n} from 'react'\nimport styled from '@emotion/styled'\nimport { AutoSizer, List } from 'react-virtualized'\nimport Box from '../../Box'\nimport FormFieldAddon from '../../FormFieldAddon'\nimport Highlighter from '../../Highlighter'\nimport TooltipPopover from '../../TooltipPopover'\nimport Icon from '../../Icon'\nimport * as Elements from './PopupSelectorElements'\nimport { OPTION_HEIGHT, SEPARATOR } from './constants'\n\nconst {\n  AddLink,\n  AddLinkPopover,\n  Header,\n  ListContainer,\n  ListItem,\n  NoResults,\n  SearchInput,\n  SearchInputContainer,\n  SearchListContainer,\n  SelectorPopover,\n  Textarea\n} = Elements\n\nexport type Value = string | number\n\nexport type Option = {\n  label: string\n  value: Value\n}\n\nexport type OptionFormatterArguments = {\n  focused: boolean\n  option: Option\n  searchString: string\n}\n\nconst DefaultOptionFormatterContainer = styled('div')({\n  whiteSpace: 'nowrap',\n  overflow: 'hidden',\n  textOverflow: 'ellipsis'\n})\n\nconst defaultOptionFormatter = ({\n  option,\n  searchString\n}: OptionFormatterArguments) => (\n  <DefaultOptionFormatterContainer title={option.label}>\n    <Highlighter textToHighlight={option.label} searchString={searchString} />\n  </DefaultOptionFormatterContainer>\n)\n\nexport type RenderContainerProps = {\n  children: ReactNode\n  triggerWidth: number\n}\n\nexport type TriggerRendererArguments = {\n  error?: Array<string>\n  getRef?: RefObject<any>\n  isOpen?: boolean\n  onClick: () => void\n  options: Array<Option>\n  tags: ReactNode\n  tooltip: ReactNode\n}\n\nconst defaultTriggerRenderer = ({\n  error,\n  getRef,\n  onClick,\n  tooltip\n}: TriggerRendererArguments) => {\n  const hasError = error && error.length > 0\n  const hasTooltip = !!tooltip\n  const trigger = (\n    <AddLink circle ref={getRef} hasError={hasError} onClick={onClick}>\n      <Icon.Small.SymbolPlus focusable=\"false\" />\n    </AddLink>\n  )\n  if (hasTooltip || hasError) {\n    return (\n      <>\n        {trigger}\n        <TooltipPopover placement=\"top\" target={getRef} interactive={false}>\n          <AddLinkPopover backgroundColor={hasError ? '#ffeef0' : ''}>\n            {tooltip}\n            {hasError && (\n              <FormFieldAddon errorList={error} truncateErrors={false} />\n            )}\n          </AddLinkPopover>\n        </TooltipPopover>\n      </>\n    )\n  }\n  return trigger\n}\n\nexport type PopupSelectorProps = {\n  creatable?: boolean\n  error?: Array<string>\n  onSelect: (value: Value[]) => void\n  onOpen: (isOpen: boolean) => void\n  isOpen?: boolean\n  key: number\n  hintText?: ReactNode\n  options: Option[]\n  optionFormatter: (formatterArgs: OptionFormatterArguments) => ReactNode\n  optionRenderContainer: (props: {\n    children: ReactNode\n    key: string\n    style: CSSProperties\n    role: string\n    hovered: boolean\n    onClick: () => void\n    onMouseEnter: () => void\n  }) => JSX.Element\n  placeholder?: string\n  positionProps?: Partial<ComponentProps<typeof TooltipPopover>>\n  renderContainer: (props: RenderContainerProps) => JSX.Element\n  showHeaders?: boolean\n  showSelectedCount?: boolean\n  tags?: ReactNode\n  triggerRenderer?: (rendererArgs: TriggerRendererArguments) => ReactNode\n  triggerTooltip?: ReactNode\n  value: Value[]\n}\n\ntype PopupSelectorState = {\n  imeComposing: boolean\n  searchString: string\n  selectedIndex: number\n  textValue: string | null\n  triggerWidth: number\n}\n\nclass PopupSelector extends PureComponent<\n  PopupSelectorProps,\n  PopupSelectorState\n> {\n  static defaultProps = {\n    optionFormatter: defaultOptionFormatter,\n    placeholder: '<Type to filter>',\n    triggerRenderer: defaultTriggerRenderer,\n    optionRenderContainer: props => <ListItem {...props} />,\n    renderContainer: ({ triggerWidth, ...props }: RenderContainerProps) => (\n      <SelectorPopover arrowHidden {...props} />\n    )\n  }\n\n  static Elements = Elements\n\n  static getDerivedStateFromProps(\n    props: PopupSelectorProps,\n    state: PopupSelectorState\n  ) {\n    if (state.textValue == null && props.value && props.options) {\n      // need to map values to labels\n      return {\n        textValue: props.value\n          .map(value => {\n            const existingOption = props.options.find(\n              option => value === option.value\n            )\n            return existingOption ? existingOption.label : value\n          })\n          // pad with another empty newline because the last empty newline is removed on close\n          .concat(props.value.length > 0 ? [''] : [])\n          .join(SEPARATOR)\n      }\n    }\n    return null\n  }\n\n  state = {\n    imeComposing: false,\n    searchString: '',\n    selectedIndex: 0,\n    textValue: null,\n    triggerWidth: 0\n  }\n\n  triggerRef = createRef<any>()\n  popoverRef = createRef<HTMLDivElement>()\n\n  componentDidMount() {\n    if (this.state.triggerWidth !== this.triggerRef.current?.offsetWidth) {\n      this.setState({\n        triggerWidth: this.triggerRef.current?.offsetWidth\n      })\n    }\n  }\n\n  componentDidUpdate() {\n    if (this.state.triggerWidth !== this.triggerRef.current?.offsetWidth) {\n      this.setState({\n        triggerWidth: this.triggerRef.current?.offsetWidth\n      })\n    }\n  }\n\n  handleSelect = (label: string) => {\n    this.setState(\n      state => ({\n        textValue: [\n          state.textValue,\n          state.textValue && state.textValue.length ? SEPARATOR : '',\n          label\n        ].join('')\n      }),\n      () => {\n        if (this.popoverRef && this.popoverRef.current) {\n          // TODO: get textarea ref so queryselector is not necessary\n          const textarea = this.popoverRef.current.querySelector(\n            'textarea[name=\"list\"]'\n          )\n          if (textarea) {\n            textarea.scrollTop = textarea.scrollHeight\n          }\n        }\n      }\n    )\n  }\n\n  handleInputKeyDown = (\n    event: KeyboardEvent<HTMLInputElement>,\n    options: Option[]\n  ) => {\n    if (this.state.imeComposing) {\n      return\n    }\n    const { selectedIndex } = this.state\n    const safeSelectedIndex = this.safeIndex(selectedIndex, options)\n    switch (event.key) {\n      case 'Tab':\n      case 'Enter':\n        event.preventDefault()\n        if (!event.shiftKey && options.length) {\n          this.handleSelect(options[safeSelectedIndex].label)\n          this.setState({ searchString: '' })\n        }\n        break\n      case 'ArrowDown':\n        event.preventDefault()\n        this.setState(state => ({\n          selectedIndex:\n            state.selectedIndex < options.length - 1\n              ? this.safeIndex(state.selectedIndex, options) + 1\n              : 0\n        }))\n        break\n      case 'ArrowUp':\n        event.preventDefault()\n        this.setState(state => ({\n          selectedIndex:\n            state.selectedIndex > 0\n              ? this.safeIndex(state.selectedIndex, options) - 1\n              : options.length - 1\n        }))\n        break\n      case 'Escape':\n        this.close()\n        if (this.triggerRef && this.triggerRef.current) {\n          this.triggerRef.current.focus()\n        }\n        break\n    }\n  }\n\n  // keep track of IME state so we know when to ignore keyDown events\n  // more info: https://developer.mozilla.org/en-US/docs/Mozilla/IME_handling_guide\n  handleImeComposing = (composing: boolean) => {\n    if (!composing) {\n      // Safari bug: compositionEnd fires BEFORE keyDown so we need to wait for keyDown to process first\n      // more info: https://medium.com/square-corner-blog/understanding-composition-browser-events-f402a8ed5643\n      window.setTimeout(\n        () => this.setState(() => ({ imeComposing: composing })),\n        10\n      )\n    } else {\n      this.setState(() => ({ imeComposing: composing }))\n    }\n  }\n\n  handleSearch = ({ value }: { value?: unknown }) => {\n    this.setState(() => ({\n      searchString: `${value}`\n    }))\n  }\n\n  close = () => {\n    this.props.onSelect(this.textToValue())\n    this.props.onOpen(false)\n  }\n\n  filteredOptions = () => {\n    const { creatable, options, value } = this.props\n    const { searchString, textValue = '' }: PopupSelectorState = this.state\n    const selectedValues = textValue ? textValue.split(SEPARATOR) : []\n    const unselectedOptions: Option[] = options.filter(({ label }) => {\n      return !selectedValues.includes(String(label))\n    })\n\n    if (searchString) {\n      let filteredOptions: Option[] = unselectedOptions.filter(\n        ({ label }: Option) =>\n          String(label).toLowerCase().includes(searchString.toLowerCase())\n      )\n\n      if (creatable) {\n        const newValue = searchString.trim()\n        const alreadyInValue = value.includes(newValue)\n        const alreadyAnOption = unselectedOptions.some(\n          option => option.value === newValue\n        )\n        if (!alreadyInValue && !alreadyAnOption) {\n          const newOption: Option = {\n            label: newValue,\n            value: newValue\n          }\n\n          filteredOptions = [newOption].concat(filteredOptions)\n        }\n      }\n\n      return filteredOptions\n    }\n\n    return unselectedOptions\n  }\n\n  textToValue(): Value[] {\n    const { options } = this.props\n    const { textValue } = this.state\n    // text contains labels so we need to convert to values if option exists\n    const selectedOptions = (textValue || '').split(SEPARATOR).map(label => {\n      const existingOption = options.find(option => label === option.label)\n      return existingOption ? existingOption.value : label\n    })\n    if (selectedOptions[selectedOptions.length - 1] === '') {\n      // remove last empty newline\n      return selectedOptions.slice(0, selectedOptions.length - 1)\n    }\n    return selectedOptions\n  }\n\n  safeIndex = (index: number, options: Option[]) => {\n    return Math.max(Math.min(index, options.length - 1), 0)\n  }\n\n  renderList = (filteredOptions: Option[], selectedIndex: number) => {\n    const { optionFormatter, optionRenderContainer } = this.props\n    const { searchString } = this.state\n    return (\n      <ListContainer>\n        <AutoSizer>\n          {({ height, width }) => (\n            <List\n              rowRenderer={({ key, index, style }) =>\n                optionRenderContainer({\n                  key,\n                  style,\n                  role: 'option',\n                  children: optionFormatter({\n                    focused: selectedIndex === index,\n                    option: filteredOptions[index],\n                    searchString\n                  }),\n                  hovered: selectedIndex === index,\n                  onClick: () => {\n                    this.handleSelect(filteredOptions[index].label)\n                    if (this.popoverRef && this.popoverRef.current) {\n                      // TODO: get input ref so queryselector is not necessary\n                      const input: HTMLInputElement | null =\n                        this.popoverRef.current.querySelector(\n                          'input[name=\"search\"]'\n                        )\n                      if (input) {\n                        input.focus()\n                      }\n                      if (filteredOptions.length === 1) {\n                        this.setState({ searchString: '' })\n                      }\n                    }\n                  },\n                  onMouseEnter: () =>\n                    this.setState(() => ({ selectedIndex: index }))\n                })\n              }\n              noRowsRenderer={() => <NoResults>No results found</NoResults>}\n              height={height}\n              width={width}\n              rowHeight={OPTION_HEIGHT}\n              rowCount={filteredOptions.length}\n              role=\"listbox\"\n              scrollToIndex={selectedIndex}\n              tabIndex={-1}\n            />\n          )}\n        </AutoSizer>\n      </ListContainer>\n    )\n  }\n\n  renderOverlay = () => {\n    const {\n      placeholder,\n      creatable,\n      hintText,\n      renderContainer,\n      showSelectedCount,\n      showHeaders\n    } = this.props\n    const { selectedIndex, textValue } = this.state\n    const filteredOptions = this.filteredOptions()\n    const safeSelectedIndex = this.safeIndex(selectedIndex, filteredOptions)\n\n    return renderContainer({\n      triggerWidth: this.state.triggerWidth,\n      children: (\n        <div onClick={event => event.stopPropagation()} ref={this.popoverRef}>\n          <Box direction=\"row\">\n            <Box grow={1}>\n              {showHeaders && <Header h4>Select Value(s):</Header>}\n              <SearchListContainer>\n                <SearchInputContainer>\n                  <SearchInput\n                    autoFocus\n                    autoComplete=\"off\"\n                    name=\"search\"\n                    value={this.state.searchString}\n                    onChange={this.handleSearch}\n                    onCompositionStart={() => this.handleImeComposing(true)}\n                    onCompositionEnd={() => this.handleImeComposing(false)}\n                    onKeyDown={event =>\n                      this.handleInputKeyDown(event, filteredOptions)\n                    }\n                    placeholder={placeholder || 'Search'}\n                  />\n                </SearchInputContainer>\n                {this.renderList(filteredOptions, safeSelectedIndex)}\n              </SearchListContainer>\n            </Box>\n            <Box grow={1}>\n              {showHeaders && <Header h4>Selected Value(s):</Header>}\n              <Textarea\n                name=\"list\"\n                type=\"textarea\"\n                wrap=\"off\"\n                onKeyPress={\n                  !creatable\n                    ? event => {\n                        // prevent text entry\n                        event.preventDefault()\n                      }\n                    : undefined\n                }\n                onChange={({ value }) => {\n                  this.setState({\n                    textValue: value === undefined ? null : `${value}`\n                  })\n                }}\n                placeholder={\n                  creatable\n                    ? 'Select values from the left\\nOR\\nType directly / paste from clipboard here.\\n\\nOne value per row.'\n                    : 'Select values from the left.'\n                }\n                value={textValue || ''}\n              />\n            </Box>\n          </Box>\n          {(hintText || showSelectedCount) && (\n            <Box\n              direction=\"row\"\n              justify=\"between\"\n              css={theme => ({\n                padding: `${theme.space[2]} ${theme.space[4]}`,\n                fontSize: theme.fontSize[0],\n                color: theme.palette.neutral[10]\n              })}\n            >\n              {hintText && <Box>{hintText}</Box>}\n              {showSelectedCount && (\n                <Box>{this.textToValue().length} selected values</Box>\n              )}\n            </Box>\n          )}\n        </div>\n      )\n    })\n  }\n\n  render() {\n    const {\n      isOpen,\n      onOpen,\n      triggerRenderer = defaultTriggerRenderer,\n      triggerTooltip,\n      error,\n      options,\n      positionProps,\n      tags\n    } = this.props\n    return (\n      <>\n        {triggerRenderer({\n          error,\n          getRef: this.triggerRef,\n          isOpen,\n          onClick: () => onOpen(true),\n          options,\n          tags,\n          tooltip: triggerTooltip || null\n        })}\n        <TooltipPopover\n          arrowHidden\n          onHide={this.close}\n          show={isOpen}\n          target={this.triggerRef}\n          placement=\"bottom\"\n          popperOptions={{\n            modifiers: [\n              {\n                name: 'preventOverflow',\n                options: {\n                  altAxis: true,\n                  mainAxis: false\n                }\n              }\n            ]\n          }}\n          {...positionProps}\n        >\n          {this.renderOverlay()}\n        </TooltipPopover>\n      </>\n    )\n  }\n}\n\nexport default PopupSelector\n"]} */",
67
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/Tagger/components/PopupSelector.tsx"],"names":[],"mappings":"AAgDwC","file":"../../../../src/Tagger/components/PopupSelector.tsx","sourcesContent":["/* eslint-disable td/instrumentation */\n// this level of fine grained instrumentation is not needed\nimport {\n  ReactNode,\n  RefObject,\n  CSSProperties,\n  ComponentProps,\n  KeyboardEvent,\n  createRef,\n  PureComponent\n} from 'react'\nimport styled from '@emotion/styled'\nimport { AutoSizer, List } from 'react-virtualized'\nimport Box from '../../Box'\nimport FormFieldAddon from '../../FormFieldAddon'\nimport Highlighter from '../../Highlighter'\nimport TooltipPopover from '../../TooltipPopover'\nimport Icon from '../../Icon'\nimport * as Elements from './PopupSelectorElements'\nimport { OPTION_HEIGHT, SEPARATOR } from './constants'\n\nconst {\n  AddLink,\n  AddLinkPopover,\n  Header,\n  ListContainer,\n  ListItem,\n  NoResults,\n  SearchInput,\n  SearchInputContainer,\n  SearchListContainer,\n  SelectorPopover,\n  Textarea\n} = Elements\n\nexport type Value = string | number\n\nexport type Option = {\n  label: string\n  value: Value\n}\n\nexport type OptionFormatterArguments = {\n  focused: boolean\n  option: Option\n  searchString: string\n}\n\nconst DefaultOptionFormatterContainer = styled('div')({\n  whiteSpace: 'nowrap',\n  overflow: 'hidden',\n  textOverflow: 'ellipsis'\n})\n\nconst defaultOptionFormatter = ({\n  option,\n  searchString\n}: OptionFormatterArguments) => (\n  <DefaultOptionFormatterContainer title={option.label}>\n    <Highlighter textToHighlight={option.label} searchString={searchString} />\n  </DefaultOptionFormatterContainer>\n)\n\nexport type RenderContainerProps = {\n  children: ReactNode\n  triggerWidth: number\n}\n\nexport type TriggerRendererArguments = {\n  error?: Array<string>\n  getRef?: RefObject<any>\n  isOpen?: boolean\n  onClick: () => void\n  options: Array<Option>\n  tags: ReactNode\n  tooltip: ReactNode\n}\n\nconst defaultTriggerRenderer = ({\n  error,\n  getRef,\n  onClick,\n  tooltip\n}: TriggerRendererArguments) => {\n  const hasError = error && error.length > 0\n  const hasTooltip = !!tooltip\n  const trigger = (\n    <AddLink circle ref={getRef} hasError={hasError} onClick={onClick}>\n      <Icon.Small.SymbolPlus focusable=\"false\" />\n    </AddLink>\n  )\n  if (hasTooltip || hasError) {\n    return (\n      <>\n        {trigger}\n        <TooltipPopover placement=\"top\" target={getRef} interactive={false}>\n          <AddLinkPopover backgroundColor={hasError ? '#ffeef0' : ''}>\n            {tooltip}\n            {hasError && (\n              <FormFieldAddon errorList={error} truncateErrors={false} />\n            )}\n          </AddLinkPopover>\n        </TooltipPopover>\n      </>\n    )\n  }\n  return trigger\n}\n\nexport type PopupSelectorProps = {\n  creatable?: boolean\n  error?: Array<string>\n  onSelect: (value: Value[]) => void\n  onOpen: (isOpen: boolean) => void\n  isOpen?: boolean\n  key: number\n  hintText?: ReactNode\n  options: Option[]\n  optionFormatter: (formatterArgs: OptionFormatterArguments) => ReactNode\n  optionRenderContainer: (props: {\n    children: ReactNode\n    key: string\n    style: CSSProperties\n    role: string\n    hovered: boolean\n    onClick: () => void\n    onMouseEnter: () => void\n  }) => JSX.Element\n  placeholder?: string\n  positionProps?: Partial<ComponentProps<typeof TooltipPopover>>\n  renderContainer: (props: RenderContainerProps) => JSX.Element\n  showHeaders?: boolean\n  showSelectedCount?: boolean\n  tags?: ReactNode\n  triggerRenderer?: (rendererArgs: TriggerRendererArguments) => ReactNode\n  triggerTooltip?: ReactNode\n  value: Value[]\n}\n\ntype PopupSelectorState = {\n  imeComposing: boolean\n  searchString: string\n  selectedIndex: number\n  textValue: string | null\n  triggerWidth: number\n}\n\nconst getNewEntry = (textValue, label) => {\n  return [\n    // Trim last separator (\\n) to avoid adding it unintentionally\n    textValue.endsWith(SEPARATOR)\n      ? textValue.slice(0, -SEPARATOR.length)\n      : textValue,\n    textValue && textValue.length ? SEPARATOR : '',\n    label\n  ].join('')\n}\n\nclass PopupSelector extends PureComponent<\n  PopupSelectorProps,\n  PopupSelectorState\n> {\n  static defaultProps = {\n    optionFormatter: defaultOptionFormatter,\n    placeholder: '<Type to filter>',\n    triggerRenderer: defaultTriggerRenderer,\n    optionRenderContainer: props => <ListItem {...props} />,\n    renderContainer: ({ triggerWidth, ...props }: RenderContainerProps) => (\n      <SelectorPopover arrowHidden {...props} />\n    )\n  }\n\n  static Elements = Elements\n\n  static getDerivedStateFromProps(\n    props: PopupSelectorProps,\n    state: PopupSelectorState\n  ) {\n    if (state.textValue == null && props.value && props.options) {\n      // need to map values to labels\n      return {\n        textValue: props.value\n          .map(value => {\n            const existingOption = props.options.find(\n              option => value === option.value\n            )\n            return existingOption ? existingOption.label : value\n          })\n          // pad with another empty newline because the last empty newline is removed on close\n          .concat(props.value.length > 0 ? [''] : [])\n          .join(SEPARATOR)\n      }\n    }\n    return null\n  }\n\n  state = {\n    imeComposing: false,\n    searchString: '',\n    selectedIndex: 0,\n    textValue: null,\n    triggerWidth: 0\n  }\n\n  triggerRef = createRef<any>()\n  popoverRef = createRef<HTMLDivElement>()\n\n  componentDidMount() {\n    if (this.state.triggerWidth !== this.triggerRef.current?.offsetWidth) {\n      this.setState({\n        triggerWidth: this.triggerRef.current?.offsetWidth\n      })\n    }\n  }\n\n  componentDidUpdate() {\n    if (this.state.triggerWidth !== this.triggerRef.current?.offsetWidth) {\n      this.setState({\n        triggerWidth: this.triggerRef.current?.offsetWidth\n      })\n    }\n  }\n\n  handleSelect = (label: string) => {\n    this.setState(\n      state => ({\n        textValue: getNewEntry(state.textValue, label)\n      }),\n      () => {\n        if (this.popoverRef && this.popoverRef.current) {\n          // TODO: get textarea ref so queryselector is not necessary\n          const textarea = this.popoverRef.current.querySelector(\n            'textarea[name=\"list\"]'\n          )\n          if (textarea) {\n            textarea.scrollTop = textarea.scrollHeight\n          }\n        }\n      }\n    )\n  }\n\n  handleInputKeyDown = (\n    event: KeyboardEvent<HTMLInputElement>,\n    options: Option[]\n  ) => {\n    if (this.state.imeComposing) {\n      return\n    }\n    const { selectedIndex } = this.state\n    const safeSelectedIndex = this.safeIndex(selectedIndex, options)\n    switch (event.key) {\n      case 'Tab':\n      case 'Enter':\n        event.preventDefault()\n        if (!event.shiftKey && options.length) {\n          this.handleSelect(options[safeSelectedIndex].label)\n          this.setState({ searchString: '' })\n        }\n        break\n      case 'ArrowDown':\n        event.preventDefault()\n        this.setState(state => ({\n          selectedIndex:\n            state.selectedIndex < options.length - 1\n              ? this.safeIndex(state.selectedIndex, options) + 1\n              : 0\n        }))\n        break\n      case 'ArrowUp':\n        event.preventDefault()\n        this.setState(state => ({\n          selectedIndex:\n            state.selectedIndex > 0\n              ? this.safeIndex(state.selectedIndex, options) - 1\n              : options.length - 1\n        }))\n        break\n      case 'Escape':\n        this.close()\n        if (this.triggerRef && this.triggerRef.current) {\n          this.triggerRef.current.focus()\n        }\n        break\n    }\n  }\n\n  // keep track of IME state so we know when to ignore keyDown events\n  // more info: https://developer.mozilla.org/en-US/docs/Mozilla/IME_handling_guide\n  handleImeComposing = (composing: boolean) => {\n    if (!composing) {\n      // Safari bug: compositionEnd fires BEFORE keyDown so we need to wait for keyDown to process first\n      // more info: https://medium.com/square-corner-blog/understanding-composition-browser-events-f402a8ed5643\n      window.setTimeout(\n        () => this.setState(() => ({ imeComposing: composing })),\n        10\n      )\n    } else {\n      this.setState(() => ({ imeComposing: composing }))\n    }\n  }\n\n  handleSearch = ({ value }: { value?: unknown }) => {\n    this.setState(() => ({\n      searchString: `${value}`\n    }))\n  }\n\n  close = () => {\n    this.props.onSelect(this.textToValue())\n    this.props.onOpen(false)\n  }\n\n  filteredOptions = () => {\n    const { creatable, options, value } = this.props\n    const { searchString, textValue = '' }: PopupSelectorState = this.state\n    const selectedValues = textValue ? textValue.split(SEPARATOR) : []\n    const unselectedOptions: Option[] = options.filter(({ label }) => {\n      return !selectedValues.includes(String(label))\n    })\n\n    if (searchString) {\n      let filteredOptions: Option[] = unselectedOptions.filter(\n        ({ label }: Option) =>\n          String(label).toLowerCase().includes(searchString.toLowerCase())\n      )\n\n      if (creatable) {\n        const newValue = searchString.trim()\n        const alreadyInValue = value.includes(newValue)\n        const alreadyAnOption = unselectedOptions.some(\n          option => option.value === newValue\n        )\n        if (!alreadyInValue && !alreadyAnOption) {\n          const newOption: Option = {\n            label: newValue,\n            value: newValue\n          }\n\n          filteredOptions = [newOption].concat(filteredOptions)\n        }\n      }\n\n      return filteredOptions\n    }\n\n    return unselectedOptions\n  }\n\n  textToValue(): Value[] {\n    const { options } = this.props\n    const { textValue } = this.state\n    // text contains labels so we need to convert to values if option exists\n    const selectedOptions = (textValue || '').split(SEPARATOR).map(label => {\n      const existingOption = options.find(option => label === option.label)\n      return existingOption ? existingOption.value : label\n    })\n\n    if (selectedOptions[selectedOptions.length - 1] === '') {\n      // remove last empty newline\n      return selectedOptions.slice(0, selectedOptions.length - 1)\n    }\n    return selectedOptions\n  }\n\n  safeIndex = (index: number, options: Option[]) => {\n    return Math.max(Math.min(index, options.length - 1), 0)\n  }\n\n  renderList = (filteredOptions: Option[], selectedIndex: number) => {\n    const { optionFormatter, optionRenderContainer } = this.props\n    const { searchString } = this.state\n    return (\n      <ListContainer>\n        <AutoSizer>\n          {({ height, width }) => (\n            <List\n              rowRenderer={({ key, index, style }) =>\n                optionRenderContainer({\n                  key,\n                  style,\n                  role: 'option',\n                  children: optionFormatter({\n                    focused: selectedIndex === index,\n                    option: filteredOptions[index],\n                    searchString\n                  }),\n                  hovered: selectedIndex === index,\n                  onClick: () => {\n                    this.handleSelect(filteredOptions[index].label)\n                    if (this.popoverRef && this.popoverRef.current) {\n                      // TODO: get input ref so queryselector is not necessary\n                      const input: HTMLInputElement | null =\n                        this.popoverRef.current.querySelector(\n                          'input[name=\"search\"]'\n                        )\n                      if (input) {\n                        input.focus()\n                      }\n                      if (filteredOptions.length === 1) {\n                        this.setState({ searchString: '' })\n                      }\n                    }\n                  },\n                  onMouseEnter: () =>\n                    this.setState(() => ({ selectedIndex: index }))\n                })\n              }\n              noRowsRenderer={() => <NoResults>No results found</NoResults>}\n              height={height}\n              width={width}\n              rowHeight={OPTION_HEIGHT}\n              rowCount={filteredOptions.length}\n              role=\"listbox\"\n              scrollToIndex={selectedIndex}\n              tabIndex={-1}\n            />\n          )}\n        </AutoSizer>\n      </ListContainer>\n    )\n  }\n\n  renderOverlay = () => {\n    const {\n      placeholder,\n      creatable,\n      hintText,\n      renderContainer,\n      showSelectedCount,\n      showHeaders\n    } = this.props\n    const { selectedIndex, textValue } = this.state\n    const filteredOptions = this.filteredOptions()\n    const safeSelectedIndex = this.safeIndex(selectedIndex, filteredOptions)\n\n    return renderContainer({\n      triggerWidth: this.state.triggerWidth,\n      children: (\n        <div onClick={event => event.stopPropagation()} ref={this.popoverRef}>\n          <Box direction=\"row\">\n            <Box grow={1}>\n              {showHeaders && <Header h4>Select Value(s):</Header>}\n              <SearchListContainer>\n                <SearchInputContainer>\n                  <SearchInput\n                    autoFocus\n                    autoComplete=\"off\"\n                    name=\"search\"\n                    value={this.state.searchString}\n                    onChange={this.handleSearch}\n                    onCompositionStart={() => this.handleImeComposing(true)}\n                    onCompositionEnd={() => this.handleImeComposing(false)}\n                    onKeyDown={event =>\n                      this.handleInputKeyDown(event, filteredOptions)\n                    }\n                    placeholder={placeholder || 'Search'}\n                  />\n                </SearchInputContainer>\n                {this.renderList(filteredOptions, safeSelectedIndex)}\n              </SearchListContainer>\n            </Box>\n            <Box grow={1}>\n              {showHeaders && <Header h4>Selected Value(s):</Header>}\n              <Textarea\n                name=\"list\"\n                type=\"textarea\"\n                wrap=\"off\"\n                onKeyPress={\n                  !creatable\n                    ? event => {\n                        // prevent text entry\n                        event.preventDefault()\n                      }\n                    : undefined\n                }\n                onChange={({ value }) => {\n                  this.setState({\n                    textValue: value === undefined ? null : `${value}`\n                  })\n                }}\n                placeholder={\n                  creatable\n                    ? 'Select values from the left\\nOR\\nType directly / paste from clipboard here.\\n\\nOne value per row.'\n                    : 'Select values from the left.'\n                }\n                value={textValue || ''}\n              />\n            </Box>\n          </Box>\n          {(hintText || showSelectedCount) && (\n            <Box\n              direction=\"row\"\n              justify=\"between\"\n              css={theme => ({\n                padding: `${theme.space[2]} ${theme.space[4]}`,\n                fontSize: theme.fontSize[0],\n                color: theme.palette.neutral[10]\n              })}\n            >\n              {hintText && <Box>{hintText}</Box>}\n              {showSelectedCount && (\n                <Box>{this.textToValue().length} selected values</Box>\n              )}\n            </Box>\n          )}\n        </div>\n      )\n    })\n  }\n\n  render() {\n    const {\n      isOpen,\n      onOpen,\n      triggerRenderer = defaultTriggerRenderer,\n      triggerTooltip,\n      error,\n      options,\n      positionProps,\n      tags\n    } = this.props\n    return (\n      <>\n        {triggerRenderer({\n          error,\n          getRef: this.triggerRef,\n          isOpen,\n          onClick: () => onOpen(true),\n          options,\n          tags,\n          tooltip: triggerTooltip || null\n        })}\n        <TooltipPopover\n          arrowHidden\n          onHide={this.close}\n          show={isOpen}\n          target={this.triggerRef}\n          placement=\"bottom\"\n          popperOptions={{\n            modifiers: [\n              {\n                name: 'preventOverflow',\n                options: {\n                  altAxis: true,\n                  mainAxis: false\n                }\n              }\n            ]\n          }}\n          {...positionProps}\n        >\n          {this.renderOverlay()}\n        </TooltipPopover>\n      </>\n    )\n  }\n}\n\nexport default PopupSelector\n"]} */",
68
68
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
69
69
  });
70
70
  var defaultOptionFormatter = function defaultOptionFormatter(_ref) {
@@ -121,6 +121,11 @@ var defaultTriggerRenderer = function defaultTriggerRenderer(_ref2) {
121
121
  }
122
122
  return trigger;
123
123
  };
124
+ var getNewEntry = function getNewEntry(textValue, label) {
125
+ return [
126
+ // Trim last separator (\n) to avoid adding it unintentionally
127
+ textValue.endsWith(_constants.SEPARATOR) ? textValue.slice(0, -_constants.SEPARATOR.length) : textValue, textValue && textValue.length ? _constants.SEPARATOR : '', label].join('');
128
+ };
124
129
  var PopupSelector = /*#__PURE__*/function (_PureComponent) {
125
130
  function PopupSelector() {
126
131
  var _this;
@@ -141,7 +146,7 @@ var PopupSelector = /*#__PURE__*/function (_PureComponent) {
141
146
  _defineProperty(_this, "handleSelect", function (label) {
142
147
  _this.setState(function (state) {
143
148
  return {
144
- textValue: [state.textValue, state.textValue && state.textValue.length ? _constants.SEPARATOR : '', label].join('')
149
+ textValue: getNewEntry(state.textValue, label)
145
150
  };
146
151
  }, function () {
147
152
  if (_this.popoverRef && _this.popoverRef.current) {
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "td-stylekit",
3
- "version": "30.4.0",
3
+ "version": "30.4.2",
4
4
  "main": "dist/es/index.js",
5
5
  "module": "dist/es/index.js",
6
6
  "types": "dist/es/index.d.ts",