td-stylekit 30.3.0 → 30.4.1
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.1](https://github.com/treasure-data/td-stylekit/compare/v30.4.0...v30.4.1) (2024-11-28)
|
|
2
|
+
|
|
3
|
+
|
|
4
|
+
### Bug Fixes
|
|
5
|
+
|
|
6
|
+
* **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))
|
|
7
|
+
|
|
8
|
+
# [30.4.0](https://github.com/treasure-data/td-stylekit/compare/v30.3.0...v30.4.0) (2024-11-26)
|
|
9
|
+
|
|
10
|
+
|
|
11
|
+
### Features
|
|
12
|
+
|
|
13
|
+
* **CON-17506:** Add merge icon ([#1615](https://github.com/treasure-data/td-stylekit/issues/1615)) ([1eba284](https://github.com/treasure-data/td-stylekit/commit/1eba28473510d76d0d7d5f17fbb629ff78c1a6fc))
|
|
14
|
+
|
|
1
15
|
# [30.3.0](https://github.com/treasure-data/td-stylekit/compare/v30.2.0...v30.3.0) (2024-11-26)
|
|
2
16
|
|
|
3
17
|
|
|
@@ -162,6 +162,8 @@ export type IconTypes = {
|
|
|
162
162
|
MasterSegmentIcon: React.ComponentType<JSX.IntrinsicElements['svg']>;
|
|
163
163
|
Megaphone: React.ComponentType<JSX.IntrinsicElements['svg']>;
|
|
164
164
|
MegaphoneIcon: React.ComponentType<JSX.IntrinsicElements['svg']>;
|
|
165
|
+
Merge: React.ComponentType<JSX.IntrinsicElements['svg']>;
|
|
166
|
+
MergeIcon: React.ComponentType<JSX.IntrinsicElements['svg']>;
|
|
165
167
|
Mixin: React.ComponentType<JSX.IntrinsicElements['svg']>;
|
|
166
168
|
MixinIcon: React.ComponentType<JSX.IntrinsicElements['svg']>;
|
|
167
169
|
More: React.ComponentType<JSX.IntrinsicElements['svg']>;
|
package/dist/es/Icon/types.js
CHANGED
|
@@ -297,6 +297,8 @@ export const types = {typesv5: {Activation: /*#__PURE__*/React.createElement("pa
|
|
|
297
297
|
})),Megaphone: /*#__PURE__*/React.createElement("path", {
|
|
298
298
|
d: "M18.126 14.107A3.2 3.2 0 0 0 20 11.194v-.8a3.2 3.2 0 0 0-3.2-3.2h-2.4l-.103.001-8.058-3.06A1.6 1.6 0 0 0 4 5.602v10.383a1.6 1.6 0 0 0 2.24 1.467L12 15.393v2.453a2.333 2.333 0 0 0 4.532.777zm-3.462-5.313v4H16.8a1.6 1.6 0 0 0 1.6-1.6v-.8a1.6 1.6 0 0 0-1.6-1.6zm-.264 5.6-.102-.002-.698.304v3.15a.733.733 0 0 0 1.423.244l1.305-3.696z",
|
|
299
299
|
fillRule: "evenodd"
|
|
300
|
+
}),Merge: /*#__PURE__*/React.createElement("path", {
|
|
301
|
+
d: "M18.234 13.002a1.33 1.33 0 0 0-.295-1.447l-3.331-3.33a1.33 1.33 0 0 0-1.882 1.881l1.063 1.063h-3.457a2.67 2.67 0 0 1-2.67-2.67V5.83A1.33 1.33 0 0 0 5 5.83v2.667c0 1.594.7 3.025 1.808 4.002A5.32 5.32 0 0 0 5 16.502v2.667a1.33 1.33 0 1 0 2.661 0v-2.667a2.67 2.67 0 0 1 2.671-2.671h3.457l-1.063 1.063a1.33 1.33 0 0 0 1.882 1.882l3.334-3.335c.13-.13.227-.28.292-.44"
|
|
300
302
|
}),Mixin: /*#__PURE__*/React.createElement("path", {
|
|
301
303
|
d: "M8 4h4a.8.8 0 0 1 .8.8v2.4a.8.8 0 0 1-.8.8h-1.245v2.4H12a.8.8 0 0 1 .8.8v.44h1.6v-.44a.8.8 0 0 1 .8-.8h4a.8.8 0 0 1 .8.8v2.4a.8.8 0 0 1-.8.8h-1.261v2.997c0 .69-.56 1.25-1.25 1.25H9.654v.309a1.15 1.15 0 0 1-1.15 1.15h-3.61a1.15 1.15 0 0 1-1.15-1.15v-2.147c0-.636.515-1.15 1.15-1.15h3.61c.635 0 1.15.515 1.15 1.15v.338h6.785V14.4H15.2a.8.8 0 0 1-.8-.8v-.46h-1.6v.46a.8.8 0 0 1-.8.8H8a.8.8 0 0 1-.8-.8v-2.4a.8.8 0 0 1 .8-.8h1.255V8H8a.8.8 0 0 1-.8-.8V4.8A.8.8 0 0 1 8 4M5.044 18.806v-1.847h3.31v1.847z",
|
|
302
304
|
fillRule: "evenodd"
|
|
@@ -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:
|
|
149
|
+
textValue: getNewEntry(state.textValue, label)
|
|
145
150
|
};
|
|
146
151
|
}, function () {
|
|
147
152
|
if (_this.popoverRef && _this.popoverRef.current) {
|