From d50f960a675941d8052bae473003895c837a737a Mon Sep 17 00:00:00 2001 From: Jan-Gerke Salomon Date: Wed, 6 May 2020 13:07:57 +0200 Subject: [PATCH 1/3] refactor(transfer): align with select & monorepo structure This commit does several things: * Use strings as selected values * Use strings to store which options are highlighted * Removes jest test abstractions * Simplifies internal names * Replaces `common.js` files with files whose names match their content * Moves the Transfer component from `core` to `widgets` * Adds the `.js` extension to all import statements where applicable * Provides the correct `disabled` & `dataTest` props to the re-order up button's icon BREAKING CHANGE: The Transfer component now expects strings as selected values instead of option objects. BREAKING CHANGE: The Transfer component is now part of `widgets` --- .../Transfer/display-order/index.js | 12 +- packages/core/src/Transfer/Filter.js | 40 -- packages/core/src/Transfer/PickedOptions.js | 59 --- packages/core/src/Transfer/SourceOptions.js | 60 --- packages/core/src/Transfer/__e2e__/common.js | 140 ------- .../__e2e__/display-order.stories.e2e.js | 32 -- .../highlight-range-of-options.stories.e2e.js | 44 --- .../src/Transfer/__tests__/common.test.js | 275 -------------- .../__tests__/common/createChildren.js | 9 - .../addAllSelectableSourceOptions.test.js | 64 ---- .../helper/addIndividualSourceOptions.test.js | 105 ----- .../helper/extractPickedReactOptions.test.js | 79 ---- .../__tests__/helper/filterOutOptions.test.js | 24 -- .../helper/filterReactOptionsBy.test.js | 38 -- .../getPlainOptionFromReactOption.test.js | 33 -- .../getPlainOptionsFromReactOptions.test.js | 60 --- .../helper/getSubsetByFilter.test.js | 81 ---- .../helper/isReorderDownDisabled.test.js | 54 --- .../helper/isReorderUpDisabled.test.js | 54 --- .../moveHighlightedPickedOptionUp.test.js | 57 --- packages/core/src/Transfer/common.js | 113 ------ .../Transfer/helper/defaultFilterCallback.js | 11 - .../helper/extractPickedReactOptions.js | 33 -- .../src/Transfer/helper/filterOutOptions.js | 13 - .../Transfer/helper/filterReactOptionsBy.js | 12 - .../helper/getPlainOptionFromReactOption.js | 12 - .../helper/getPlainOptionsFromReactOptions.js | 10 - .../src/Transfer/helper/getSubsetByFilter.js | 29 -- .../Transfer/helper/isReorderDownDisabled.js | 17 - .../Transfer/helper/isReorderUpDisabled.js | 17 - .../helper/moveHighlightedPickedOptionDown.js | 33 -- .../helper/moveHighlightedPickedOptionUp.js | 32 -- .../helper/useHighlightedOptions/toggleAdd.js | 21 - .../useHighlightedOptions/toggleReplace.js | 23 -- packages/core/src/Transfer/types.js | 12 - packages/core/src/index.js | 2 - .../{core => widgets}/src/Transfer/Actions.js | 0 .../{core => widgets}/src/Transfer/AddAll.js | 2 +- .../src/Transfer/AddIndividual.js | 2 +- .../src/Transfer/Container.js | 0 packages/widgets/src/Transfer/Filter.js | 35 ++ .../src/Transfer/LeftFooter.js | 2 +- .../src/Transfer/LeftHeader.js | 2 +- .../src/Transfer/LeftSide.js | 2 +- .../widgets/src/Transfer/PickedOptions.js | 29 ++ .../src/Transfer/RemoveAll.js | 2 +- .../src/Transfer/RemoveIndividual.js | 2 +- .../src/Transfer/ReorderingActions.js | 9 +- .../src/Transfer/RightFooter.js | 2 +- .../src/Transfer/RightSide.js | 2 +- .../widgets/src/Transfer/SourceOptions.js | 29 ++ .../src/Transfer/Transfer.js | 204 +++++----- .../src/Transfer/Transfer.stories.js | 359 ++++++++---------- .../addAllSelectableSourceOptions.js | 27 +- .../Transfer}/addIndividualSourceOptions.js | 38 +- .../Transfer}/createDoubleClickHandlers.js | 16 +- .../Transfer/defaultFilterCallback.js | 9 + .../Transfer/getOptionClickHandlers.js | 19 + .../src/Transfer/Transfer}/index.js | 5 +- .../Transfer/isReorderDownDisabled.js | 11 + .../Transfer/Transfer/isReorderUpDisabled.js | 11 + .../moveHighlightedPickedOptionDown.js | 29 ++ .../Transfer/moveHighlightedPickedOptionUp.js | 29 ++ .../Transfer}/removeAllPickedOptions.js | 0 .../removeIndividualPickedOptions.js | 16 +- .../Transfer}/useHighlightedOptions.js | 14 +- .../createToggleHighlightedOption.js | 27 +- .../useHighlightedOptions/toggleAdd.js | 20 + .../useHighlightedOptions/toggleRange.js | 26 +- .../useHighlightedOptions/toggleReplace.js | 26 ++ .../src/Transfer/TransferOption.js | 17 +- ..._remove-highlighted-options.stories.e2e.js | 18 +- .../widgets/src/Transfer/__e2e__/common.js | 0 .../src/Transfer/__e2e__/common/options.js | 92 +++++ .../__e2e__/common/statefulDecorator.js | 20 + .../disabled-transfer-buttons.stories.e2e.js | 29 +- .../disabled-transfer-options.stories.e2e.js | 13 +- .../__e2e__/display-order.stories.e2e.js | 26 ++ .../filter-options-list.stories.e2e.js | 33 +- .../highlight-range-of-options.stories.e2e.js | 54 +++ .../reorder-with-buttons.stories.e2e.js | 14 +- ...et_unset-highlighted-option.stories.e2e.js | 18 +- .../__e2e__/transferring-items.stories.e2e.js | 18 +- .../src/Transfer/__tests__/common.test.js | 131 +++++++ .../addAllSelectableSourceOptions.test.js | 46 +++ .../helper/addIndividualSourceOptions.test.js | 80 ++++ .../helper/defaultFilterCallback.test.js | 12 +- .../helper/isReorderDownDisabled.test.js | 47 +++ .../helper/isReorderUpDisabled.test.js | 47 +++ .../moveHighlightedPickedOptionDown.test.js | 33 +- .../moveHighlightedPickedOptionUp.test.js | 48 +++ .../helper/removeAllPickedOptions.test.js | 2 +- .../removeIndividualPickedOptions.test.js | 24 +- .../createToggleHighlightedOption.test.js | 25 +- .../useHighlightedOption/toggleAdd.test.js | 20 +- .../useHighlightedOption/toggleRange.test.js | 80 ++-- .../toggleReplace.test.js | 8 +- .../helper/useHighlightedOptions.test.js | 6 +- .../widgets/src/Transfer/common/addOption.js | 12 + .../src/Transfer/common/findOptionIndex.js | 9 + .../Transfer/common/getModeByModifierKey.js | 29 ++ packages/widgets/src/Transfer/common/index.js | 6 + .../widgets/src/Transfer/common/isOption.js | 7 + packages/widgets/src/Transfer/common/modes.js | 11 + .../src/Transfer/common/removeOption.js | 15 + .../widgets/src/Transfer/common/styles.js | 4 + .../src/Transfer/common/toggleValue.js | 18 + .../{core => widgets}/src/Transfer/icons.js | 0 packages/widgets/src/index.js | 2 + 109 files changed, 1501 insertions(+), 2289 deletions(-) delete mode 100644 packages/core/src/Transfer/Filter.js delete mode 100644 packages/core/src/Transfer/PickedOptions.js delete mode 100644 packages/core/src/Transfer/SourceOptions.js delete mode 100644 packages/core/src/Transfer/__e2e__/common.js delete mode 100644 packages/core/src/Transfer/__e2e__/display-order.stories.e2e.js delete mode 100644 packages/core/src/Transfer/__e2e__/highlight-range-of-options.stories.e2e.js delete mode 100644 packages/core/src/Transfer/__tests__/common.test.js delete mode 100644 packages/core/src/Transfer/__tests__/common/createChildren.js delete mode 100644 packages/core/src/Transfer/__tests__/helper/addAllSelectableSourceOptions.test.js delete mode 100644 packages/core/src/Transfer/__tests__/helper/addIndividualSourceOptions.test.js delete mode 100644 packages/core/src/Transfer/__tests__/helper/extractPickedReactOptions.test.js delete mode 100644 packages/core/src/Transfer/__tests__/helper/filterOutOptions.test.js delete mode 100644 packages/core/src/Transfer/__tests__/helper/filterReactOptionsBy.test.js delete mode 100644 packages/core/src/Transfer/__tests__/helper/getPlainOptionFromReactOption.test.js delete mode 100644 packages/core/src/Transfer/__tests__/helper/getPlainOptionsFromReactOptions.test.js delete mode 100644 packages/core/src/Transfer/__tests__/helper/getSubsetByFilter.test.js delete mode 100644 packages/core/src/Transfer/__tests__/helper/isReorderDownDisabled.test.js delete mode 100644 packages/core/src/Transfer/__tests__/helper/isReorderUpDisabled.test.js delete mode 100644 packages/core/src/Transfer/__tests__/helper/moveHighlightedPickedOptionUp.test.js delete mode 100644 packages/core/src/Transfer/common.js delete mode 100644 packages/core/src/Transfer/helper/defaultFilterCallback.js delete mode 100644 packages/core/src/Transfer/helper/extractPickedReactOptions.js delete mode 100644 packages/core/src/Transfer/helper/filterOutOptions.js delete mode 100644 packages/core/src/Transfer/helper/filterReactOptionsBy.js delete mode 100644 packages/core/src/Transfer/helper/getPlainOptionFromReactOption.js delete mode 100644 packages/core/src/Transfer/helper/getPlainOptionsFromReactOptions.js delete mode 100644 packages/core/src/Transfer/helper/getSubsetByFilter.js delete mode 100644 packages/core/src/Transfer/helper/isReorderDownDisabled.js delete mode 100644 packages/core/src/Transfer/helper/isReorderUpDisabled.js delete mode 100644 packages/core/src/Transfer/helper/moveHighlightedPickedOptionDown.js delete mode 100644 packages/core/src/Transfer/helper/moveHighlightedPickedOptionUp.js delete mode 100644 packages/core/src/Transfer/helper/useHighlightedOptions/toggleAdd.js delete mode 100644 packages/core/src/Transfer/helper/useHighlightedOptions/toggleReplace.js delete mode 100644 packages/core/src/Transfer/types.js rename packages/{core => widgets}/src/Transfer/Actions.js (100%) rename packages/{core => widgets}/src/Transfer/AddAll.js (93%) rename packages/{core => widgets}/src/Transfer/AddIndividual.js (93%) rename packages/{core => widgets}/src/Transfer/Container.js (100%) create mode 100644 packages/widgets/src/Transfer/Filter.js rename packages/{core => widgets}/src/Transfer/LeftFooter.js (91%) rename packages/{core => widgets}/src/Transfer/LeftHeader.js (91%) rename packages/{core => widgets}/src/Transfer/LeftSide.js (93%) create mode 100644 packages/widgets/src/Transfer/PickedOptions.js rename packages/{core => widgets}/src/Transfer/RemoveAll.js (93%) rename packages/{core => widgets}/src/Transfer/RemoveIndividual.js (93%) rename packages/{core => widgets}/src/Transfer/ReorderingActions.js (91%) rename packages/{core => widgets}/src/Transfer/RightFooter.js (91%) rename packages/{core => widgets}/src/Transfer/RightSide.js (92%) create mode 100644 packages/widgets/src/Transfer/SourceOptions.js rename packages/{core => widgets}/src/Transfer/Transfer.js (69%) rename packages/{core => widgets}/src/Transfer/Transfer.stories.js (51%) rename packages/{core/src/Transfer/helper => widgets/src/Transfer/Transfer}/addAllSelectableSourceOptions.js (54%) rename packages/{core/src/Transfer/helper => widgets/src/Transfer/Transfer}/addIndividualSourceOptions.js (60%) rename packages/{core/src/Transfer/helper => widgets/src/Transfer/Transfer}/createDoubleClickHandlers.js (72%) create mode 100644 packages/widgets/src/Transfer/Transfer/defaultFilterCallback.js create mode 100644 packages/widgets/src/Transfer/Transfer/getOptionClickHandlers.js rename packages/{core/src/Transfer/helper => widgets/src/Transfer/Transfer}/index.js (74%) create mode 100644 packages/widgets/src/Transfer/Transfer/isReorderDownDisabled.js create mode 100644 packages/widgets/src/Transfer/Transfer/isReorderUpDisabled.js create mode 100644 packages/widgets/src/Transfer/Transfer/moveHighlightedPickedOptionDown.js create mode 100644 packages/widgets/src/Transfer/Transfer/moveHighlightedPickedOptionUp.js rename packages/{core/src/Transfer/helper => widgets/src/Transfer/Transfer}/removeAllPickedOptions.js (100%) rename packages/{core/src/Transfer/helper => widgets/src/Transfer/Transfer}/removeIndividualPickedOptions.js (58%) rename packages/{core/src/Transfer/helper => widgets/src/Transfer/Transfer}/useHighlightedOptions.js (79%) rename packages/{core/src/Transfer/helper => widgets/src/Transfer/Transfer}/useHighlightedOptions/createToggleHighlightedOption.js (55%) create mode 100644 packages/widgets/src/Transfer/Transfer/useHighlightedOptions/toggleAdd.js rename packages/{core/src/Transfer/helper => widgets/src/Transfer/Transfer}/useHighlightedOptions/toggleRange.js (72%) create mode 100644 packages/widgets/src/Transfer/Transfer/useHighlightedOptions/toggleReplace.js rename packages/{core => widgets}/src/Transfer/TransferOption.js (90%) rename packages/{core => widgets}/src/Transfer/__e2e__/add_remove-highlighted-options.stories.e2e.js (53%) create mode 100644 packages/widgets/src/Transfer/__e2e__/common.js create mode 100644 packages/widgets/src/Transfer/__e2e__/common/options.js create mode 100644 packages/widgets/src/Transfer/__e2e__/common/statefulDecorator.js rename packages/{core => widgets}/src/Transfer/__e2e__/disabled-transfer-buttons.stories.e2e.js (52%) rename packages/{core => widgets}/src/Transfer/__e2e__/disabled-transfer-options.stories.e2e.js (58%) create mode 100644 packages/widgets/src/Transfer/__e2e__/display-order.stories.e2e.js rename packages/{core => widgets}/src/Transfer/__e2e__/filter-options-list.stories.e2e.js (86%) create mode 100644 packages/widgets/src/Transfer/__e2e__/highlight-range-of-options.stories.e2e.js rename packages/{core => widgets}/src/Transfer/__e2e__/reorder-with-buttons.stories.e2e.js (50%) rename packages/{core => widgets}/src/Transfer/__e2e__/set_unset-highlighted-option.stories.e2e.js (53%) rename packages/{core => widgets}/src/Transfer/__e2e__/transferring-items.stories.e2e.js (51%) create mode 100644 packages/widgets/src/Transfer/__tests__/common.test.js create mode 100644 packages/widgets/src/Transfer/__tests__/helper/addAllSelectableSourceOptions.test.js create mode 100644 packages/widgets/src/Transfer/__tests__/helper/addIndividualSourceOptions.test.js rename packages/{core => widgets}/src/Transfer/__tests__/helper/defaultFilterCallback.test.js (70%) create mode 100644 packages/widgets/src/Transfer/__tests__/helper/isReorderDownDisabled.test.js create mode 100644 packages/widgets/src/Transfer/__tests__/helper/isReorderUpDisabled.test.js rename packages/{core => widgets}/src/Transfer/__tests__/helper/moveHighlightedPickedOptionDown.test.js (50%) create mode 100644 packages/widgets/src/Transfer/__tests__/helper/moveHighlightedPickedOptionUp.test.js rename packages/{core => widgets}/src/Transfer/__tests__/helper/removeAllPickedOptions.test.js (88%) rename packages/{core => widgets}/src/Transfer/__tests__/helper/removeIndividualPickedOptions.test.js (60%) rename packages/{core => widgets}/src/Transfer/__tests__/helper/useHighlightedOption/createToggleHighlightedOption.test.js (79%) rename packages/{core => widgets}/src/Transfer/__tests__/helper/useHighlightedOption/toggleAdd.test.js (78%) rename packages/{core => widgets}/src/Transfer/__tests__/helper/useHighlightedOption/toggleRange.test.js (65%) rename packages/{core => widgets}/src/Transfer/__tests__/helper/useHighlightedOption/toggleReplace.test.js (77%) rename packages/{core => widgets}/src/Transfer/__tests__/helper/useHighlightedOptions.test.js (84%) create mode 100644 packages/widgets/src/Transfer/common/addOption.js create mode 100644 packages/widgets/src/Transfer/common/findOptionIndex.js create mode 100644 packages/widgets/src/Transfer/common/getModeByModifierKey.js create mode 100644 packages/widgets/src/Transfer/common/index.js create mode 100644 packages/widgets/src/Transfer/common/isOption.js create mode 100644 packages/widgets/src/Transfer/common/modes.js create mode 100644 packages/widgets/src/Transfer/common/removeOption.js create mode 100644 packages/widgets/src/Transfer/common/styles.js create mode 100644 packages/widgets/src/Transfer/common/toggleValue.js rename packages/{core => widgets}/src/Transfer/icons.js (100%) diff --git a/cypress/integration/Transfer/display-order/index.js b/cypress/integration/Transfer/display-order/index.js index 23eb64c90a..a077264c34 100644 --- a/cypress/integration/Transfer/display-order/index.js +++ b/cypress/integration/Transfer/display-order/index.js @@ -61,9 +61,9 @@ When('the user selects multiple options', () => { }) .each($option => cy.wrap($option).clickWith('ctrl')) .then($options => { - cy.wrap($options.toArray().map(extractOptionFromElement)) - .as('selectedOptions') - .then(console.log.bind(null, 'selectedOptions')) + cy.wrap($options.toArray().map(extractOptionFromElement)).as( + 'selectedOptions' + ) }) cy.get('{transfer-actions-addindividual}').click() @@ -150,7 +150,6 @@ Then( () => cy.get('{transfer-sourceoptions} {transferoption}'), () => cy.get('@deselectedOptions') ).should(([win, $selectableSourceOptions, deselectedOptions]) => { - console.log('deselectedOptions', deselectedOptions) const selectableSourceOptions = $selectableSourceOptions .toArray() .map(extractOptionFromElement) @@ -169,9 +168,6 @@ Then( value === deselectedOption.value ) - if (!result) { - console.log('deselectedOption', deselectedOption) - } return result }) expect(hasAllOptions).to.equal(true) @@ -205,8 +201,6 @@ Then( .slice(transferredOptions.length * -1) .map(extractOptionFromElement) - console.log('transferredOptions', transferredOptions) - console.log('lastSelectedOptions', lastSelectedOptions) expect(transferredOptions).to.eql(lastSelectedOptions) }) } diff --git a/packages/core/src/Transfer/Filter.js b/packages/core/src/Transfer/Filter.js deleted file mode 100644 index c8bbf91afc..0000000000 --- a/packages/core/src/Transfer/Filter.js +++ /dev/null @@ -1,40 +0,0 @@ -import React from 'react' -import propTypes from '@dhis2/prop-types' - -import { spacers } from '@dhis2/ui-constants' - -import { Input } from '../Input/Input.js' -import { Field } from '../Field/Field.js' - -export const Filter = ({ dataTest, filter, onChange, label }) => { - return ( -
- - - - - -
- ) -} - -Filter.propTypes = { - dataTest: propTypes.string.isRequired, - filter: propTypes.string.isRequired, - onChange: propTypes.func.isRequired, - label: propTypes.string, -} diff --git a/packages/core/src/Transfer/PickedOptions.js b/packages/core/src/Transfer/PickedOptions.js deleted file mode 100644 index d0218f9484..0000000000 --- a/packages/core/src/Transfer/PickedOptions.js +++ /dev/null @@ -1,59 +0,0 @@ -import React from 'react' -import propTypes from '@dhis2/prop-types' - -import { spacers } from '@dhis2/ui-constants' - -import { findOption, getModeByModifierKey } from './common.js' - -// TODO: This will be refactored away to match the MultiSelect -export const multiSelectedPropType = propTypes.arrayOf( - propTypes.shape({ - label: propTypes.string, - value: propTypes.string, - }) -) - -export const PickedOptions = ({ - children, - dataTest, - toggleHighlightedPickedOption, - selectedEmptyComponent, - highlightedPickedOptions, - deselectSingleOption, -}) => ( -
- {!React.Children.count(children) && selectedEmptyComponent} - {React.Children.map(children, child => { - const option = { - label: child.props.label, - value: child.props.value, - } - - return React.cloneElement(child, { - onClick: (_, event) => { - const mode = getModeByModifierKey(event) - toggleHighlightedPickedOption({ option, mode }) - }, - onDoubleClick: deselectSingleOption, - highlighted: !!findOption(highlightedPickedOptions, option), - }) - })} - - -
-) - -PickedOptions.propTypes = { - children: propTypes.node.isRequired, - dataTest: propTypes.string.isRequired, - deselectSingleOption: propTypes.func.isRequired, - toggleHighlightedPickedOption: propTypes.func.isRequired, - highlightedPickedOptions: multiSelectedPropType, - selectedEmptyComponent: propTypes.node, -} diff --git a/packages/core/src/Transfer/SourceOptions.js b/packages/core/src/Transfer/SourceOptions.js deleted file mode 100644 index af4a0e1810..0000000000 --- a/packages/core/src/Transfer/SourceOptions.js +++ /dev/null @@ -1,60 +0,0 @@ -import React from 'react' -import propTypes from '@dhis2/prop-types' - -import { spacers } from '@dhis2/ui-constants' - -import { findOption, getModeByModifierKey } from './common.js' - -// TODO: This will be refactored away to match the MultiSelect -export const multiSelectedPropType = propTypes.arrayOf( - propTypes.shape({ - label: propTypes.string, - value: propTypes.string, - }) -) - -export const SourceOptions = ({ - children, - dataTest, - toggleHighlightedSourceOption, - sourceEmptyPlaceholder, - highlightedSourceOptions, - selectSingleOption, -}) => ( -
- {React.Children.map(children, child => { - const option = { - label: child.props.label, - value: child.props.value, - } - - return React.cloneElement(child, { - onClick: (_, event) => { - const mode = getModeByModifierKey(event) - toggleHighlightedSourceOption({ option, mode }) - }, - onDoubleClick: selectSingleOption, - highlighted: !!findOption(highlightedSourceOptions, option), - }) - })} - - {!React.Children.count(children) && sourceEmptyPlaceholder} - - -
-) - -SourceOptions.propTypes = { - dataTest: propTypes.string.isRequired, - selectSingleOption: propTypes.func.isRequired, - toggleHighlightedSourceOption: propTypes.func.isRequired, - children: propTypes.node, - highlightedSourceOptions: multiSelectedPropType, - sourceEmptyPlaceholder: propTypes.node, -} diff --git a/packages/core/src/Transfer/__e2e__/common.js b/packages/core/src/Transfer/__e2e__/common.js deleted file mode 100644 index 9c8f57b50d..0000000000 --- a/packages/core/src/Transfer/__e2e__/common.js +++ /dev/null @@ -1,140 +0,0 @@ -import React, { useState } from 'react' -import { TransferOption } from '../../index.js' - -export const extractOptionFromComponent = ({ props }) => ({ - label: props.label, - value: props.value, -}) - -export const statefulDecorator = ({ - initialState = [], - controlFilter = false, - initialSearchTerm = '', -} = {}) => fn => - React.createElement(() => { - const initialSelected = initialState.map(child => child.props) - const [selected, setSelected] = useState(initialSelected) - const [searchTerm, setSearchTerm] = useState(initialSearchTerm) - - return fn({ - selected, - searchTerm: controlFilter ? searchTerm : undefined, - onChange: payload => setSelected(payload.selected), - onFilterChange: controlFilter - ? ({ value }) => setSearchTerm(value) - : undefined, - }) - }) - -export const options = [ - , - , - , - , - , - , - , - , - , - , - , - , - , - , - , - , - , - , - , - , - , - , -] diff --git a/packages/core/src/Transfer/__e2e__/display-order.stories.e2e.js b/packages/core/src/Transfer/__e2e__/display-order.stories.e2e.js deleted file mode 100644 index abe2c0338f..0000000000 --- a/packages/core/src/Transfer/__e2e__/display-order.stories.e2e.js +++ /dev/null @@ -1,32 +0,0 @@ -/* eslint-disable react/prop-types */ -import React from 'react' - -import { Transfer } from '../../index.js' -import { options, statefulDecorator } from './common.js' - -export default { title: 'Transfer Display Order' } - -window.options = options.map(child => { - const { label, value } = child.props - return { label, value } -}) - -export const NoSelection = ({ selected, onChange }) => ( - - {options} - -) - -export const SomeSelected = ({ selected, onChange }) => ( - - {options} - -) - -SomeSelected.story = { - decorators: [ - statefulDecorator({ - initialState: options.slice(0, 4), - }), - ], -} diff --git a/packages/core/src/Transfer/__e2e__/highlight-range-of-options.stories.e2e.js b/packages/core/src/Transfer/__e2e__/highlight-range-of-options.stories.e2e.js deleted file mode 100644 index 855d09349f..0000000000 --- a/packages/core/src/Transfer/__e2e__/highlight-range-of-options.stories.e2e.js +++ /dev/null @@ -1,44 +0,0 @@ -/* eslint-disable react/prop-types */ -import React from 'react' - -import { Transfer } from '../../index.js' -import { options, statefulDecorator } from './common.js' - -export default { - title: 'Transfer highlight range of options', - decorators: [statefulDecorator()], -} - -export const HasOptions = ({ onChange, selected }) => ( - - {options} - -) - -export const HasSelected = ({ onChange, selected }) => ( - - {options} - -) - -HasSelected.story = { - decorators: [ - statefulDecorator({ - initialState: options.slice(0, 4), - }), - ], -} - -export const AllSelected = ({ onChange, selected }) => ( - - {options} - -) - -AllSelected.story = { - decorators: [ - statefulDecorator({ - initialState: options, - }), - ], -} diff --git a/packages/core/src/Transfer/__tests__/common.test.js b/packages/core/src/Transfer/__tests__/common.test.js deleted file mode 100644 index 8c33215dee..0000000000 --- a/packages/core/src/Transfer/__tests__/common.test.js +++ /dev/null @@ -1,275 +0,0 @@ -import { - addOption, - isOption, - findOption, - findOptionIndex, - removeOption, - toggleOption, - toggleOptions, -} from '../../Transfer/common.js' - -describe('Transfer - isOption', () => { - it('should return true when the options are the same', () => { - const option1 = { label: 'foo', value: 'bar' } - const option2 = { label: 'foo', value: 'bar' } - const actual = isOption(option1, option2) - const expected = true - - expect(actual).toBe(expected) - }) - - it('should return false when the labels do not match', () => { - const option1 = { label: 'foo', value: 'bar' } - const option2 = { label: 'baz', value: 'bar' } - const actual = isOption(option1, option2) - const expected = false - - expect(actual).toBe(expected) - }) - - it('should return false when the values do not match', () => { - const option1 = { label: 'foo', value: 'bar' } - const option2 = { label: 'foo', value: 'baz' } - const actual = isOption(option1, option2) - const expected = false - - expect(actual).toBe(expected) - }) -}) - -describe('Transfer - findOptionIndex', () => { - it('should return index 1', () => { - const options = [ - { label: 'foo', value: 'bar' }, - { label: 'foo', value: 'baz' }, - ] - const option = { label: 'foo', value: 'baz' } - const actual = findOptionIndex(options, option) - const expected = 1 - - expect(actual).toBe(expected) - }) - - it('should return -1 when the option is not included', () => { - const options = [ - { label: 'foo', value: 'bar' }, - { label: 'foo', value: 'baz' }, - ] - const option = { label: 'baz', value: 'baz' } - const actual = findOptionIndex(options, option) - const expected = -1 - - expect(actual).toBe(expected) - }) -}) - -describe('Transfer - findOption', () => { - it('should return the option', () => { - const options = [ - { label: 'foo', value: 'bar' }, - { label: 'foo', value: 'baz' }, - ] - const option = { label: 'foo', value: 'baz' } - const actual = findOption(options, option) - const expected = { label: 'foo', value: 'baz' } - - expect(actual).toEqual(expected) - }) - - it('should return undefined when the option is not included', () => { - const options = [ - { label: 'foo', value: 'bar' }, - { label: 'foo', value: 'baz' }, - ] - const option = { label: 'baz', value: 'baz' } - const actual = findOption(options, option) - const expected = undefined - - expect(actual).toEqual(expected) - }) -}) - -describe('Transfer - toggleOption', () => { - it('should add the option to the collection', () => { - const options = [ - { label: 'foo', value: 'bar' }, - { label: 'foo', value: 'baz' }, - ] - const option = { label: 'baz', value: 'baz' } - const actual = toggleOption(options, option) - const expected = [ - { label: 'foo', value: 'bar' }, - { label: 'foo', value: 'baz' }, - { label: 'baz', value: 'baz' }, - ] - - expect(actual).toEqual(expected) - }) - - it('should remove the option from the collection', () => { - const options = [ - { label: 'foo', value: 'bar' }, - { label: 'foo', value: 'baz' }, - { label: 'baz', value: 'baz' }, - ] - const option = { label: 'baz', value: 'baz' } - const actual = toggleOption(options, option) - const expected = [ - { label: 'foo', value: 'bar' }, - { label: 'foo', value: 'baz' }, - ] - - expect(actual).toEqual(expected) - }) -}) - -describe('Transfer - addOption', () => { - it('should add the option to the collection', () => { - const options = [ - { label: 'foo', value: 'bar' }, - { label: 'foo', value: 'baz' }, - ] - const option = { label: 'baz', value: 'baz' } - const actual = addOption(options, option) - const expected = [ - { label: 'foo', value: 'bar' }, - { label: 'foo', value: 'baz' }, - { label: 'baz', value: 'baz' }, - ] - - expect(actual).toEqual(expected) - }) - - it('should not add the option to the collection when already included', () => { - const options = [ - { label: 'foo', value: 'bar' }, - { label: 'foo', value: 'baz' }, - { label: 'baz', value: 'baz' }, - ] - const option = { label: 'baz', value: 'baz' } - const actual = addOption(options, option) - const expected = [ - { label: 'foo', value: 'bar' }, - { label: 'foo', value: 'baz' }, - { label: 'baz', value: 'baz' }, - ] - - expect(actual).toEqual(expected) - }) -}) - -describe('Transfer - removeOption', () => { - it('should remove the option to the collection', () => { - const options = [ - { label: 'foo', value: 'bar' }, - { label: 'foo', value: 'baz' }, - { label: 'baz', value: 'baz' }, - ] - const option = { label: 'baz', value: 'baz' } - const actual = removeOption(options, option) - const expected = [ - { label: 'foo', value: 'bar' }, - { label: 'foo', value: 'baz' }, - ] - - expect(actual).toEqual(expected) - }) - - it('should not remove the option to the collection when already included', () => { - const options = [ - { label: 'foo', value: 'bar' }, - { label: 'foo', value: 'baz' }, - ] - const option = { label: 'baz', value: 'baz' } - const actual = removeOption(options, option) - const expected = [ - { label: 'foo', value: 'bar' }, - { label: 'foo', value: 'baz' }, - ] - - expect(actual).toEqual(expected) - }) -}) - -describe('Transfer - toggleOptions', () => { - it('should add all options when none are inside the colleciton yet', () => { - const collection = [ - { label: 'foo', value: 'foo' }, - { label: 'bar', value: 'bar' }, - ] - const optionsToToggle = [ - { label: 'baz', value: 'baz' }, - { label: 'foobar', value: 'foobar' }, - ] - const actual = toggleOptions(collection, optionsToToggle) - const expected = [ - { label: 'foo', value: 'foo' }, - { label: 'bar', value: 'bar' }, - { label: 'baz', value: 'baz' }, - { label: 'foobar', value: 'foobar' }, - ] - - expect(actual).toEqual(expected) - }) - - it('should remove all options when all are included', () => { - const collection = [ - { label: 'foo', value: 'foo' }, - { label: 'bar', value: 'bar' }, - { label: 'baz', value: 'baz' }, - { label: 'foobar', value: 'foobar' }, - ] - const optionsToToggle = [ - { label: 'baz', value: 'baz' }, - { label: 'foobar', value: 'foobar' }, - ] - const actual = toggleOptions(collection, optionsToToggle) - const expected = [ - { label: 'foo', value: 'foo' }, - { label: 'bar', value: 'bar' }, - ] - - expect(actual).toEqual(expected) - }) - - it('should add some and remove some when only some are given', () => { - const collection = [ - { label: 'foo', value: 'foo' }, - { label: 'bar', value: 'bar' }, - { label: 'baz', value: 'baz' }, - ] - const optionsToToggle = [ - { label: 'baz', value: 'baz' }, - { label: 'foobar', value: 'foobar' }, - ] - const actual = toggleOptions(collection, optionsToToggle) - const expected = [ - { label: 'foo', value: 'foo' }, - { label: 'bar', value: 'bar' }, - { label: 'foobar', value: 'foobar' }, - ] - - expect(actual).toEqual(expected) - }) - - it('should only add the missing one and not remove any when using a custom modifier', () => { - const collection = [ - { label: 'foo', value: 'foo' }, - { label: 'bar', value: 'bar' }, - { label: 'baz', value: 'baz' }, - ] - const optionsToToggle = [ - { label: 'baz', value: 'baz' }, - { label: 'foobar', value: 'foobar' }, - ] - const actual = toggleOptions(collection, optionsToToggle, addOption) - const expected = [ - { label: 'foo', value: 'foo' }, - { label: 'bar', value: 'bar' }, - { label: 'baz', value: 'baz' }, - { label: 'foobar', value: 'foobar' }, - ] - - expect(actual).toEqual(expected) - }) -}) diff --git a/packages/core/src/Transfer/__tests__/common/createChildren.js b/packages/core/src/Transfer/__tests__/common/createChildren.js deleted file mode 100644 index 3ef92d8119..0000000000 --- a/packages/core/src/Transfer/__tests__/common/createChildren.js +++ /dev/null @@ -1,9 +0,0 @@ -import { createElement } from 'react' - -export const createChildren = (...childElements) => { - const div = createElement('div', {}, childElements) - const { props } = div - const { children } = props - - return children -} diff --git a/packages/core/src/Transfer/__tests__/helper/addAllSelectableSourceOptions.test.js b/packages/core/src/Transfer/__tests__/helper/addAllSelectableSourceOptions.test.js deleted file mode 100644 index 0345c3e2d3..0000000000 --- a/packages/core/src/Transfer/__tests__/helper/addAllSelectableSourceOptions.test.js +++ /dev/null @@ -1,64 +0,0 @@ -import { createElement } from 'react' - -import { addAllSelectableSourceOptions } from '../../helper/addAllSelectableSourceOptions.js' -import { createChildren } from '../common/createChildren.js' - -describe('Transfer - addAllSelectableSourceOptions', () => { - const onChange = jest.fn() - const setHighlightedSourceOptions = jest.fn() - - const sourceReactOptions = createChildren( - createElement('div', { key: 'foo', label: 'Foo', value: 'foo' }), - createElement('div', { key: 'bar', label: 'Bar', value: 'bar' }), - createElement('div', { key: 'baz', label: 'Baz', value: 'baz' }), - createElement('div', { - key: 'foobar', - label: 'Foobar', - value: 'foobar', - }), - createElement('div', { - key: 'foobaz', - label: 'Foobaz', - value: 'foobaz', - }) - ) - - afterEach(() => { - onChange.mockClear() - setHighlightedSourceOptions.mockClear() - }) - - it('should add all selectable source options to the selectedPlainOptions array', () => { - const selectedPlainOptions = [{ label: 'Barfoo', value: 'barfoo' }] - const expected = { - selected: [ - { label: 'Barfoo', value: 'barfoo' }, - { label: 'Foo', value: 'foo' }, - { label: 'Bar', value: 'bar' }, - { label: 'Baz', value: 'baz' }, - { label: 'Foobar', value: 'foobar' }, - { label: 'Foobaz', value: 'foobaz' }, - ], - } - - addAllSelectableSourceOptions({ - sourceReactOptions, - selectedPlainOptions, - onChange, - setHighlightedSourceOptions, - }) - - expect(onChange).toHaveBeenCalledWith(expected) - }) - - it('should reset all highlighted source options', () => { - addAllSelectableSourceOptions({ - sourceReactOptions, - selectedPlainOptions: [{ label: 'Barfoo', value: 'barfoo' }], - onChange, - setHighlightedSourceOptions, - }) - - expect(setHighlightedSourceOptions).toHaveBeenCalledWith([]) - }) -}) diff --git a/packages/core/src/Transfer/__tests__/helper/addIndividualSourceOptions.test.js b/packages/core/src/Transfer/__tests__/helper/addIndividualSourceOptions.test.js deleted file mode 100644 index 9962b1a81a..0000000000 --- a/packages/core/src/Transfer/__tests__/helper/addIndividualSourceOptions.test.js +++ /dev/null @@ -1,105 +0,0 @@ -import { createElement } from 'react' - -import { addIndividualSourceOptions } from '../../helper/addIndividualSourceOptions.js' -import { createChildren } from '../common/createChildren.js' - -describe('Transfer - addIndividualSourceOptions', () => { - const onChange = jest.fn() - const setHighlightedSourceOptions = jest.fn() - - const filteredSourcePlainOptions = createChildren( - createElement('div', { key: 'foo', label: 'Foo', value: 'foo' }), - createElement('div', { - key: 'foobar', - label: 'Foobar', - value: 'foobar', - }), - createElement('div', { - key: 'foobaz', - label: 'Foobaz', - value: 'foobaz', - }) - ) - - const highlightedSourcePlainOptions = [ - { label: 'Foobaz', value: 'foobaz' }, - { label: 'Bar', value: 'bar' }, - ] - - const selectedPlainOptions = [{ label: 'Barfoo', value: 'barfoo' }] - - afterEach(() => { - onChange.mockClear() - setHighlightedSourceOptions.mockClear() - }) - - it('should add the highlighted source options to the selectedPlainOptions array', () => { - addIndividualSourceOptions({ - highlightedSourcePlainOptions, - maxSelections: Infinity, - onChange, - selectedPlainOptions, - setHighlightedSourceOptions, - }) - - expect(onChange).toHaveBeenCalledWith({ - selected: [ - { label: 'Barfoo', value: 'barfoo' }, - { label: 'Foobaz', value: 'foobaz' }, - { label: 'Bar', value: 'bar' }, - ], - }) - }) - - it('should reset the highlighted source options', () => { - addIndividualSourceOptions({ - highlightedSourcePlainOptions, - maxSelections: Infinity, - onChange, - selectedPlainOptions, - setHighlightedSourceOptions, - }) - - expect(setHighlightedSourceOptions).toHaveBeenCalledWith([]) - }) - - it('should only select the filtered source options', () => { - addIndividualSourceOptions({ - filterable: true, - filter: 'oo', - filteredSourcePlainOptions, - highlightedSourcePlainOptions, - maxSelections: Infinity, - onChange, - selectedPlainOptions, - setHighlightedSourceOptions, - }) - - expect(onChange).toHaveBeenCalledWith({ - selected: [ - { label: 'Barfoo', value: 'barfoo' }, - { label: 'Foobaz', value: 'foobaz' }, - ], - }) - }) - - it('should only call onChange with the max selection amount', () => { - addIndividualSourceOptions({ - filterable: true, - filter: 'oo', - filteredSourcePlainOptions, - highlightedSourcePlainOptions: highlightedSourcePlainOptions.slice( - 0, - 1 - ), - maxSelections: 1, - onChange, - selectedPlainOptions: selectedPlainOptions.slice(0, 1), - setHighlightedSourceOptions, - }) - - expect(onChange).toHaveBeenCalledWith({ - selected: [{ label: 'Foobaz', value: 'foobaz' }], - }) - }) -}) diff --git a/packages/core/src/Transfer/__tests__/helper/extractPickedReactOptions.test.js b/packages/core/src/Transfer/__tests__/helper/extractPickedReactOptions.test.js deleted file mode 100644 index ce506cf36d..0000000000 --- a/packages/core/src/Transfer/__tests__/helper/extractPickedReactOptions.test.js +++ /dev/null @@ -1,79 +0,0 @@ -import { Children, createElement } from 'react' - -import { createChildren } from '../common/createChildren.js' -import { extractPickedReactOptions } from '../../helper/extractPickedReactOptions.js' - -describe('Transfer - extractPickedReactOptions', () => { - it('should remove all non-picked options', () => { - const toggleHighlightedPickedOptions = jest.fn() - const highlightedPickedOptions = [] - const reactOptions = createChildren( - createElement('div', { key: 'foo', label: 'Foo', value: 'foo' }), - createElement('div', { key: 'bar', label: 'Bar', value: 'bar' }), - createElement('div', { key: 'baz', label: 'Baz', value: 'baz' }) - ) - const selectedPlainOptions = [{ label: 'Baz', value: 'baz' }] - const result = Children.toArray( - extractPickedReactOptions({ - reactOptions, - selectedPlainOptions, - highlightedPickedOptions, - toggleHighlightedPickedOptions, - }) - ) - - expect(result).toHaveLength(1) - expect(result[0].props).toEqual( - expect.objectContaining({ label: 'Baz', value: 'baz' }) - ) - }) - - it('sorts the picked options by the order of the "selectedPlainOptions" array', () => { - const toggleHighlightedPickedOptions = jest.fn() - const highlightedPickedOptions = [] - const reactOptions = createChildren( - createElement('div', { key: 'foo', label: 'Foo', value: 'foo' }), - createElement('div', { key: 'bar', label: 'Bar', value: 'bar' }), - createElement('div', { key: 'baz', label: 'Baz', value: 'baz' }), - createElement('div', { - key: 'foobar', - label: 'Foobar', - value: 'foobar', - }), - createElement('div', { - key: 'foobaz', - label: 'Foobaz', - value: 'foobaz', - }), - createElement('div', { - key: 'barfoo', - label: 'Barfoo', - value: 'barfoo', - }) - ) - const selectedPlainOptions = [ - { label: 'Barfoo', value: 'barfoo' }, - { label: 'Foo', value: 'foo' }, - { label: 'Foobar', value: 'foobar' }, - ] - const result = Children.toArray( - extractPickedReactOptions({ - reactOptions, - selectedPlainOptions, - highlightedPickedOptions, - toggleHighlightedPickedOptions, - }) - ) - - expect(result).toHaveLength(3) - expect(result[0].props).toEqual( - expect.objectContaining({ label: 'Barfoo', value: 'barfoo' }) - ) - expect(result[1].props).toEqual( - expect.objectContaining({ label: 'Foo', value: 'foo' }) - ) - expect(result[2].props).toEqual( - expect.objectContaining({ label: 'Foobar', value: 'foobar' }) - ) - }) -}) diff --git a/packages/core/src/Transfer/__tests__/helper/filterOutOptions.test.js b/packages/core/src/Transfer/__tests__/helper/filterOutOptions.test.js deleted file mode 100644 index 05d3f9887a..0000000000 --- a/packages/core/src/Transfer/__tests__/helper/filterOutOptions.test.js +++ /dev/null @@ -1,24 +0,0 @@ -import { Children, createElement } from 'react' - -import { createChildren } from '../common/createChildren.js' -import { filterOutOptions } from '../../helper/filterOutOptions.js' - -describe('Transfer - filterOutOptions', () => { - it('should remove all picked options', () => { - const reactOptions = createChildren( - createElement('div', { label: 'Foo', value: 'foo', key: 'foo' }), - createElement('div', { label: 'Bar', value: 'bar', key: 'bar' }), - createElement('div', { label: 'Baz', value: 'baz', key: 'baz' }) - ) - - const plainOptions = [{ label: 'Baz', value: 'baz' }] - - const result = Children.toArray( - filterOutOptions(reactOptions, plainOptions) - ) - - expect(result).toHaveLength(2) - expect(result[0].props).toEqual({ label: 'Foo', value: 'foo' }) - expect(result[1].props).toEqual({ label: 'Bar', value: 'bar' }) - }) -}) diff --git a/packages/core/src/Transfer/__tests__/helper/filterReactOptionsBy.test.js b/packages/core/src/Transfer/__tests__/helper/filterReactOptionsBy.test.js deleted file mode 100644 index b43317ae7f..0000000000 --- a/packages/core/src/Transfer/__tests__/helper/filterReactOptionsBy.test.js +++ /dev/null @@ -1,38 +0,0 @@ -import React from 'react' - -import { TransferOption } from '../../TransferOption.js' -import { createChildren } from '../common/createChildren.js' -import { filterReactOptionsBy } from '../../helper/filterReactOptionsBy.js' - -describe('Transfer - filterReactOptionsBy', () => { - it('should return all enabled options', () => { - const reactOptions = createChildren( - , - , - , - , - - ) - - const expectedChildren = createChildren( - , - , - - ) - - const actualChildren = filterReactOptionsBy( - ({ disabled }) => !disabled, - reactOptions - ) - - const expected = expectedChildren.map(child => child.props) - const actual = actualChildren.map(child => child.props) - - expect(actual).toEqual(expected) - }) -}) diff --git a/packages/core/src/Transfer/__tests__/helper/getPlainOptionFromReactOption.test.js b/packages/core/src/Transfer/__tests__/helper/getPlainOptionFromReactOption.test.js deleted file mode 100644 index 488e2e4baf..0000000000 --- a/packages/core/src/Transfer/__tests__/helper/getPlainOptionFromReactOption.test.js +++ /dev/null @@ -1,33 +0,0 @@ -import React from 'react' -import { getPlainOptionFromReactOption } from '../../../Transfer/helper/getPlainOptionFromReactOption.js' - -describe('Transfer - getPlainOptionFromReactOption', () => { - it('should extract the plain option', () => { - const reactOption = React.createElement('span', { - label: 'Foo', - value: 'foo', - disabled: false, - }) - const expected = { label: 'Foo', value: 'foo', disabled: false } - const actual = getPlainOptionFromReactOption(reactOption) - - expect(actual).toEqual(expected) - }) - - it('should merge additionalData into the option', () => { - const additionalData = { - bar: 'Bar', - baz: 'Baz', - } - const reactOption = React.createElement('span', { - label: 'Foo', - value: 'foo', - disabled: false, - additionalData, - }) - const expected = expect.objectContaining(additionalData) - const actual = getPlainOptionFromReactOption(reactOption) - - expect(actual).toEqual(expected) - }) -}) diff --git a/packages/core/src/Transfer/__tests__/helper/getPlainOptionsFromReactOptions.test.js b/packages/core/src/Transfer/__tests__/helper/getPlainOptionsFromReactOptions.test.js deleted file mode 100644 index 65b7d3e53f..0000000000 --- a/packages/core/src/Transfer/__tests__/helper/getPlainOptionsFromReactOptions.test.js +++ /dev/null @@ -1,60 +0,0 @@ -import { createElement } from 'react' - -import { createChildren } from '../common/createChildren.js' -import { getPlainOptionsFromReactOptions } from '../../helper/getPlainOptionsFromReactOptions.js' - -describe('Transfer - getPlainOptionsFromReactOptions', () => { - it('should extract the option objects from the reactOptions and return as array', () => { - const reactOptions = createChildren( - createElement('div', { - label: 'Foo', - value: 'foo', - key: 'foo', - }), - createElement('div', { - label: 'Bar', - value: 'bar', - key: 'bar', - }) - ) - - const expected = [ - { label: 'Foo', value: 'foo' }, - { label: 'Bar', value: 'bar' }, - ] - const actual = getPlainOptionsFromReactOptions(reactOptions) - - expect(actual).toEqual(expected) - }) - - it('should extract additionalData', () => { - const reactOptions = createChildren( - createElement('div', { - label: 'Foo', - value: 'foo', - key: 'foo', - additionalData: { - year: '2020', - relativePeriod: true, - }, - }), - createElement('div', { - label: 'Bar', - value: 'bar', - key: 'bar', - additionalData: { - year: '2019', - relativePeriod: false, - }, - }) - ) - - const expected = [ - { label: 'Foo', value: 'foo', year: '2020', relativePeriod: true }, - { label: 'Bar', value: 'bar', year: '2019', relativePeriod: false }, - ] - const actual = getPlainOptionsFromReactOptions(reactOptions) - - expect(actual).toEqual(expected) - }) -}) diff --git a/packages/core/src/Transfer/__tests__/helper/getSubsetByFilter.test.js b/packages/core/src/Transfer/__tests__/helper/getSubsetByFilter.test.js deleted file mode 100644 index d4b495955c..0000000000 --- a/packages/core/src/Transfer/__tests__/helper/getSubsetByFilter.test.js +++ /dev/null @@ -1,81 +0,0 @@ -import { Children, createElement } from 'react' - -import { createChildren } from '../common/createChildren.js' -import { getSubsetByFilter } from '../../helper/getSubsetByFilter.js' - -describe('Transfer - getSubsetByFilter', () => { - const filterCallback = (options, filter) => - options.filter(({ label }) => label.indexOf(filter) !== -1) - - const reactOptions = createChildren( - createElement('div', { key: 'foo', label: 'Foo', value: 'foo' }), - createElement('div', { key: 'bar', label: 'Bar', value: 'bar' }), - createElement('div', { key: 'baz', label: 'Baz', value: 'baz' }), - createElement('div', { - key: 'foobar', - label: 'Foobar', - value: 'foobar', - }), - createElement('div', { - key: 'foobaz', - label: 'Foobaz', - value: 'foobaz', - }), - createElement('div', { - key: 'barfoo', - label: 'Barfoo', - value: 'barfoo', - }) - ) - - it('should return only the react reactOptions with a matching label', () => { - const result = Children.toArray( - getSubsetByFilter({ - filterable: true, - filter: 'oo', - reactOptions, - filterCallback, - }) - ) - - expect(result).toHaveLength(4) - expect(result[0].props).toEqual( - expect.objectContaining({ label: 'Foo', value: 'foo' }) - ) - expect(result[1].props).toEqual( - expect.objectContaining({ label: 'Foobar', value: 'foobar' }) - ) - expect(result[2].props).toEqual( - expect.objectContaining({ label: 'Foobaz', value: 'foobaz' }) - ) - expect(result[3].props).toEqual( - expect.objectContaining({ label: 'Barfoo', value: 'barfoo' }) - ) - }) - - it('should return all reactOptions when filterable is false', () => { - const result = Children.toArray( - getSubsetByFilter({ - filterable: false, - filter: 'oo', - reactOptions, - filterCallback, - }) - ) - - expect(result).toHaveLength(6) - }) - - it('should return all reactOptions when the filter is empty', () => { - const result = Children.toArray( - getSubsetByFilter({ - filterable: true, - filter: '', - reactOptions, - filterCallback, - }) - ) - - expect(result).toHaveLength(6) - }) -}) diff --git a/packages/core/src/Transfer/__tests__/helper/isReorderDownDisabled.test.js b/packages/core/src/Transfer/__tests__/helper/isReorderDownDisabled.test.js deleted file mode 100644 index 0a3dc1fc1b..0000000000 --- a/packages/core/src/Transfer/__tests__/helper/isReorderDownDisabled.test.js +++ /dev/null @@ -1,54 +0,0 @@ -import { isReorderDownDisabled } from '../../../Transfer/helper/isReorderDownDisabled.js' - -describe('Transfer - isReorderDownDisabled', () => { - const selectedPlainOptions = [ - { label: 'Foo', value: 'foo' }, - { label: 'Bar', value: 'bar' }, - { label: 'Baz', value: 'baz' }, - ] - - it('should return true when there are no highlighted picked options', () => { - const highlightedPickedPlainOptions = [] - const actual = isReorderDownDisabled({ - highlightedPickedPlainOptions, - selectedPlainOptions, - }) - - expect(actual).toBe(true) - }) - - it('should return true when there are multiple highlighted picked options', () => { - const highlightedPickedPlainOptions = [ - { label: 'Bar', value: 'bar' }, - { label: 'Baz', value: 'baz' }, - ] - const actual = isReorderDownDisabled({ - highlightedPickedPlainOptions, - selectedPlainOptions, - }) - - expect(actual).toBe(true) - }) - - it('should return true if the last picked option is highlighted', () => { - const highlightedPickedPlainOptions = [{ label: 'Baz', value: 'baz' }] - - const actual = isReorderDownDisabled({ - highlightedPickedPlainOptions, - selectedPlainOptions, - }) - - expect(actual).toBe(true) - }) - - it('should return false when one picked option is highlighted which is not the last one', () => { - const highlightedPickedPlainOptions = [{ label: 'Bar', value: 'bar' }] - - const actual = isReorderDownDisabled({ - highlightedPickedPlainOptions, - selectedPlainOptions, - }) - - expect(actual).toBe(false) - }) -}) diff --git a/packages/core/src/Transfer/__tests__/helper/isReorderUpDisabled.test.js b/packages/core/src/Transfer/__tests__/helper/isReorderUpDisabled.test.js deleted file mode 100644 index ccbdf939a0..0000000000 --- a/packages/core/src/Transfer/__tests__/helper/isReorderUpDisabled.test.js +++ /dev/null @@ -1,54 +0,0 @@ -import { isReorderUpDisabled } from '../../../Transfer/helper/isReorderUpDisabled.js' - -describe('Transfer - isReorderUpDisabled', () => { - const selectedPlainOptions = [ - { label: 'Foo', value: 'foo' }, - { label: 'Bar', value: 'bar' }, - { label: 'Baz', value: 'baz' }, - ] - - it('should return true when there are no highlighted picked options', () => { - const highlightedPickedPlainOptions = [] - const actual = isReorderUpDisabled({ - highlightedPickedPlainOptions, - selectedPlainOptions, - }) - - expect(actual).toBe(true) - }) - - it('should return true when there are multiple highlighted picked options', () => { - const highlightedPickedPlainOptions = [ - { label: 'Bar', value: 'bar' }, - { label: 'Baz', value: 'baz' }, - ] - const actual = isReorderUpDisabled({ - highlightedPickedPlainOptions, - selectedPlainOptions, - }) - - expect(actual).toBe(true) - }) - - it('should return true if the first picked option is highlighted', () => { - const highlightedPickedPlainOptions = [{ label: 'Foo', value: 'foo' }] - - const actual = isReorderUpDisabled({ - highlightedPickedPlainOptions, - selectedPlainOptions, - }) - - expect(actual).toBe(true) - }) - - it('should return false when one picked option is highlighted which is not the last one', () => { - const highlightedPickedPlainOptions = [{ label: 'Baz', value: 'baz' }] - - const actual = isReorderUpDisabled({ - highlightedPickedPlainOptions, - selectedPlainOptions, - }) - - expect(actual).toBe(false) - }) -}) diff --git a/packages/core/src/Transfer/__tests__/helper/moveHighlightedPickedOptionUp.test.js b/packages/core/src/Transfer/__tests__/helper/moveHighlightedPickedOptionUp.test.js deleted file mode 100644 index d5598b0fbc..0000000000 --- a/packages/core/src/Transfer/__tests__/helper/moveHighlightedPickedOptionUp.test.js +++ /dev/null @@ -1,57 +0,0 @@ -import { moveHighlightedPickedOptionUp } from '../../../Transfer/helper/moveHighlightedPickedOptionUp.js' - -describe('Transfer - moveHighlightedPickedOptionUp', () => { - const onChange = jest.fn() - - const selectedPlainOptions = [ - { label: 'Foo', value: 'foo' }, - { label: 'Bar', value: 'bar' }, - { label: 'Baz', value: 'baz' }, - ] - - afterEach(() => { - onChange.mockClear() - }) - - it('should move the highlighted option up', () => { - const highlighted = [{ label: 'Bar', value: 'bar' }] - - moveHighlightedPickedOptionUp({ - selectedPlainOptions, - highlightedPickedPlainOptions: highlighted, - onChange, - }) - - expect(onChange).toHaveBeenCalledWith({ - selected: [ - { label: 'Bar', value: 'bar' }, - { label: 'Foo', value: 'foo' }, - { label: 'Baz', value: 'baz' }, - ], - }) - }) - - it('should do nothing when trying to move up the first option', () => { - const highlighted = [{ label: 'Foo', value: 'foo' }] - - moveHighlightedPickedOptionUp({ - selectedPlainOptions, - highlightedPickedPlainOptions: highlighted, - onChange, - }) - - expect(onChange).toHaveBeenCalledTimes(0) - }) - - it('should do nothing when trying to move up a non-existing option', () => { - const highlighted = [{ label: 'Foobar', value: 'foobar' }] - - moveHighlightedPickedOptionUp({ - selectedPlainOptions, - highlightedPickedPlainOptions: highlighted, - onChange, - }) - - expect(onChange).toHaveBeenCalledTimes(0) - }) -}) diff --git a/packages/core/src/Transfer/common.js b/packages/core/src/Transfer/common.js deleted file mode 100644 index 55d5c86b86..0000000000 --- a/packages/core/src/Transfer/common.js +++ /dev/null @@ -1,113 +0,0 @@ -import './types.js' -import { colors } from '@dhis2/ui-constants' - -export const borderColor = colors.grey400 -export const borderRadius = '3px' - -/** - * Click modes when clicking on an option with/without - * a modifier key (ctrl, alt, cmd, shift) - */ - -// no or multiple modifier keys -export const REPLACE_MODE = 'REPLACE_MODE' -// add/remove options from selection -export const ADD_MODE = 'ADD_MODE' -// create selection range -export const RANGE_MODE = 'RANGE_MODE' - -/** - * @param {Option} left - * @param {Option} left - * @returns {bool} - */ -export const isOption = (left, right) => - left.label === right.label && left.value === right.value - -/** - * @param {Option[]} options - * @param {Option} option - * @returns {Int} - */ -export const findOptionIndex = (options, option) => - options.findIndex(current => isOption(current, option)) - -/** - * @param {Option[]} options - * @param {Option} option - * @returns {Option} - */ -export const findOption = (options, option) => - options.find(current => isOption(current, option)) - -/** - * @param {Option[]} options - * @param {Option} option - * @returns {Option} - */ -export const addOption = (options, option) => { - const found = findOption(options, option) - if (found) return options - return [...options, option] -} - -/** - * @param {Option[]} options - * @param {Option} option - * @returns {Option} - */ -export const removeOption = (options, option) => { - const index = findOptionIndex(options, option) - - if (index === -1) return options - if (index === 0) return options.slice(1) - - return [...options.slice(0, index), ...options.slice(index + 1)] -} - -/** - * @param {Option[]} options - * @param {Option} option - * @returns {Option} - */ -export const toggleOption = (options, option) => - findOption(options, option) - ? removeOption(options, option) - : addOption(options, option) - -/** - * @param {Option[]} collection - * @param {Option[]} options - * @param {Function} modifier - * @returns {Option} - */ -export const toggleOptions = ( - collection, - optionsToToggle, - modifier = toggleOption -) => { - return optionsToToggle.reduce( - (curSelected, option) => modifier(curSelected, option), - collection - ) -} - -export const getModeByModifierKey = ({ - altKey, - shiftKey, - ctrlKey, - metaKey, -}) => { - const keys = [altKey, shiftKey, ctrlKey, metaKey] - const amountKeyPressed = keys.filter(v => v) - const moreThanOneKeyPressed = amountKeyPressed.length - - if (moreThanOneKeyPressed !== 1) return REPLACE_MODE - - if (altKey || ctrlKey || metaKey) return ADD_MODE - - if (shiftKey) return RANGE_MODE - - // default to replace mode - return REPLACE_MODE -} diff --git a/packages/core/src/Transfer/helper/defaultFilterCallback.js b/packages/core/src/Transfer/helper/defaultFilterCallback.js deleted file mode 100644 index 2cb282ccad..0000000000 --- a/packages/core/src/Transfer/helper/defaultFilterCallback.js +++ /dev/null @@ -1,11 +0,0 @@ -/** - * @param {Option[]} plainOptions - * @param {string} filter - * @returns {Option[]} - */ -export const defaultFilterCallback = (plainOptions, filter) => - filter === '' - ? plainOptions - : plainOptions.filter(({ label }) => - label.match(new RegExp(filter, 'i')) - ) diff --git a/packages/core/src/Transfer/helper/extractPickedReactOptions.js b/packages/core/src/Transfer/helper/extractPickedReactOptions.js deleted file mode 100644 index d811303ef2..0000000000 --- a/packages/core/src/Transfer/helper/extractPickedReactOptions.js +++ /dev/null @@ -1,33 +0,0 @@ -import { Children } from 'react' -import { findOption, findOptionIndex } from '../common' - -/** - * @param {Object} args - * @param {ReactElement} args.reactOptions - * @param {Option[]} args.selectedPlainOptions - * @returns {ReactElement} React elements - */ -export const extractPickedReactOptions = ({ - reactOptions, - selectedPlainOptions, -}) => { - const pickedOptions = Children.toArray(reactOptions) - .map(child => { - const { props } = child - const isSelected = !!findOption(selectedPlainOptions, props) - - return isSelected ? child : null - }) - // We can ONLY do this because the reactOptions have keys - .filter(child => !!child) - - pickedOptions.sort((left, right) => { - const leftIndex = findOptionIndex(selectedPlainOptions, left.props) - const rightIndex = findOptionIndex(selectedPlainOptions, right.props) - - if (leftIndex < rightIndex) return -1 - return 1 - }) - - return pickedOptions -} diff --git a/packages/core/src/Transfer/helper/filterOutOptions.js b/packages/core/src/Transfer/helper/filterOutOptions.js deleted file mode 100644 index 4701aab1aa..0000000000 --- a/packages/core/src/Transfer/helper/filterOutOptions.js +++ /dev/null @@ -1,13 +0,0 @@ -import { Children } from 'react' -import { findOption } from '../common' - -/** - * @param {ReactElement} reactOptions - * @param {Option[]} plainOptions - * @returns {Object} React elements - */ -export const filterOutOptions = (reactOptions, plainOptions) => { - return Children.map(reactOptions, child => - findOption(plainOptions, child.props) ? null : child - ) -} diff --git a/packages/core/src/Transfer/helper/filterReactOptionsBy.js b/packages/core/src/Transfer/helper/filterReactOptionsBy.js deleted file mode 100644 index b17009cde2..0000000000 --- a/packages/core/src/Transfer/helper/filterReactOptionsBy.js +++ /dev/null @@ -1,12 +0,0 @@ -import { Children } from 'react' -import { getPlainOptionFromReactOption } from './getPlainOptionFromReactOption' - -export const filterReactOptionsBy = (callback, reactOptions) => { - return Children.map(reactOptions, child => { - const plainOption = getPlainOptionFromReactOption(child) - const keep = callback(plainOption) - - if (!keep) return null - return child - }) -} diff --git a/packages/core/src/Transfer/helper/getPlainOptionFromReactOption.js b/packages/core/src/Transfer/helper/getPlainOptionFromReactOption.js deleted file mode 100644 index f78a48e8f3..0000000000 --- a/packages/core/src/Transfer/helper/getPlainOptionFromReactOption.js +++ /dev/null @@ -1,12 +0,0 @@ -import '../types.js' - -/** - * @param {ReactElement} reactOption - * @returns {Option} plainOption - */ -export const getPlainOptionFromReactOption = reactOption => ({ - label: reactOption.props.label, - value: reactOption.props.value, - disabled: reactOption.props.disabled, - ...(reactOption.props.additionalData || {}), -}) diff --git a/packages/core/src/Transfer/helper/getPlainOptionsFromReactOptions.js b/packages/core/src/Transfer/helper/getPlainOptionsFromReactOptions.js deleted file mode 100644 index f2a1e91c97..0000000000 --- a/packages/core/src/Transfer/helper/getPlainOptionsFromReactOptions.js +++ /dev/null @@ -1,10 +0,0 @@ -import '../types.js' -import { Children } from 'react' -import { getPlainOptionFromReactOption } from './getPlainOptionFromReactOption' - -/** - * @param {ReactElement} reactOptions - * @returns {Option[]} plainOption - */ -export const getPlainOptionsFromReactOptions = reactOptions => - Children.toArray(reactOptions).map(getPlainOptionFromReactOption) diff --git a/packages/core/src/Transfer/helper/getSubsetByFilter.js b/packages/core/src/Transfer/helper/getSubsetByFilter.js deleted file mode 100644 index 5231409168..0000000000 --- a/packages/core/src/Transfer/helper/getSubsetByFilter.js +++ /dev/null @@ -1,29 +0,0 @@ -import { Children } from 'react' -import { getPlainOptionsFromReactOptions } from './getPlainOptionsFromReactOptions' -import { isOption } from '../common' - -/** - * @param {Object} args - * @param {ReactElement} args.reactOptions - * @param {string} args.filter - * @param {bool} args.filterable - * @param {Function} args.filterCallback - * @returns {Object} React elements - */ -export const getSubsetByFilter = ({ - reactOptions, - filter, - filterable, - filterCallback, -}) => { - const options = getPlainOptionsFromReactOptions(reactOptions) - - const filtered = filterable ? filterCallback(options, filter) : options - - return Children.map(reactOptions, child => { - if (!filterable) return child - if (!filtered.find(option => isOption(option, child.props))) return null - - return child - }) -} diff --git a/packages/core/src/Transfer/helper/isReorderDownDisabled.js b/packages/core/src/Transfer/helper/isReorderDownDisabled.js deleted file mode 100644 index 2a7e5ebbda..0000000000 --- a/packages/core/src/Transfer/helper/isReorderDownDisabled.js +++ /dev/null @@ -1,17 +0,0 @@ -import { findOptionIndex } from '../common' - -/** - * @param {Object} args - * @param {PlainElement} args.highlightedPickedPlainOptions - * @param {Option[]} args.selectedPlainOptions - * @returns {bool} - */ -export const isReorderDownDisabled = ({ - highlightedPickedPlainOptions, - selectedPlainOptions, -}) => - // only one item can be moved with the buttons - highlightedPickedPlainOptions.length !== 1 || - // can't move an item down if it's the last one - findOptionIndex(selectedPlainOptions, highlightedPickedPlainOptions[0]) === - selectedPlainOptions.length - 1 diff --git a/packages/core/src/Transfer/helper/isReorderUpDisabled.js b/packages/core/src/Transfer/helper/isReorderUpDisabled.js deleted file mode 100644 index c83425136d..0000000000 --- a/packages/core/src/Transfer/helper/isReorderUpDisabled.js +++ /dev/null @@ -1,17 +0,0 @@ -import { findOptionIndex } from '../common' - -/** - * @param {Object} args - * @param {PlainElement} args.highlightedPickedPlainOptions - * @param {Option[]} args.selectedPlainOptions - * @returns {bool} - */ -export const isReorderUpDisabled = ({ - highlightedPickedPlainOptions, - selectedPlainOptions, -}) => - // only one item can be moved with the buttons - highlightedPickedPlainOptions.length !== 1 || - // can't move an item up if it's the first one - findOptionIndex(selectedPlainOptions, highlightedPickedPlainOptions[0]) === - 0 diff --git a/packages/core/src/Transfer/helper/moveHighlightedPickedOptionDown.js b/packages/core/src/Transfer/helper/moveHighlightedPickedOptionDown.js deleted file mode 100644 index 324cad5a99..0000000000 --- a/packages/core/src/Transfer/helper/moveHighlightedPickedOptionDown.js +++ /dev/null @@ -1,33 +0,0 @@ -import { findOptionIndex } from '../common' - -/** - * @param {Object} args - * @param {Option[]} args.selectedPlainOptions - * @param {Option[]} args.highlightedPickedPlainOptions - * @param {Function} args.onChange - * @returns {void} - */ -export const moveHighlightedPickedOptionDown = ({ - selectedPlainOptions, - highlightedPickedPlainOptions, - onChange, -}) => { - const optionIndex = findOptionIndex( - selectedPlainOptions, - highlightedPickedPlainOptions[0] - ) - - // Can't move down last or non-existing option - if (optionIndex === -1 || optionIndex > selectedPlainOptions.length - 2) - return - - // swap with next item - const reordered = [ - ...selectedPlainOptions.slice(0, optionIndex), - selectedPlainOptions[optionIndex + 1], - selectedPlainOptions[optionIndex], - ...selectedPlainOptions.slice(optionIndex + 2), - ] - - onChange({ selected: reordered }) -} diff --git a/packages/core/src/Transfer/helper/moveHighlightedPickedOptionUp.js b/packages/core/src/Transfer/helper/moveHighlightedPickedOptionUp.js deleted file mode 100644 index 983db5cd72..0000000000 --- a/packages/core/src/Transfer/helper/moveHighlightedPickedOptionUp.js +++ /dev/null @@ -1,32 +0,0 @@ -import { findOptionIndex } from '../common' - -/** - * @param {Object} args - * @param {Option[]} args.selectedPlainOptions - * @param {Option[]} args.highlightedPickedPlainOptions - * @param {Function} args.onChange - * @returns {void} - */ -export const moveHighlightedPickedOptionUp = ({ - selectedPlainOptions, - highlightedPickedPlainOptions, - onChange, -}) => { - const optionIndex = findOptionIndex( - selectedPlainOptions, - highlightedPickedPlainOptions[0] - ) - - // Can't move up option at index 0 or non-existing option - if (optionIndex < 1) return - - // swap with previous item - const reordered = [ - ...selectedPlainOptions.slice(0, optionIndex - 1), - selectedPlainOptions[optionIndex], - selectedPlainOptions[optionIndex - 1], - ...selectedPlainOptions.slice(optionIndex + 1), - ] - - onChange({ selected: reordered }) -} diff --git a/packages/core/src/Transfer/helper/useHighlightedOptions/toggleAdd.js b/packages/core/src/Transfer/helper/useHighlightedOptions/toggleAdd.js deleted file mode 100644 index f5d2288026..0000000000 --- a/packages/core/src/Transfer/helper/useHighlightedOptions/toggleAdd.js +++ /dev/null @@ -1,21 +0,0 @@ -import '../../types.js' -import { toggleOption } from '../../common' - -/** - * @param {Object} args - * @param {Option[]} args.highlightedOptions - * @param {number} args.maxSelections - * @param {Option} args.option - * @param {Function} args.setHighlightedOption - * @returns {void} - */ -export const toggleAdd = ({ - highlightedOptions, - maxSelections, - option, - setHighlightedOptions, -}) => { - setHighlightedOptions( - toggleOption(highlightedOptions, option).slice(-1 * maxSelections) - ) -} diff --git a/packages/core/src/Transfer/helper/useHighlightedOptions/toggleReplace.js b/packages/core/src/Transfer/helper/useHighlightedOptions/toggleReplace.js deleted file mode 100644 index 2b002e0f8c..0000000000 --- a/packages/core/src/Transfer/helper/useHighlightedOptions/toggleReplace.js +++ /dev/null @@ -1,23 +0,0 @@ -import '../../types.js' -import { toggleOption } from '../../common' - -/** - * @param {Object} args - * @param {Option[]} args.highlightedOptions - * @param {Option} args.option - * @param {Function} args.setHighlightedOption - * @returns {void} - */ -export const toggleReplace = ({ - option, - highlightedOptions, - setHighlightedOptions, -}) => { - if (highlightedOptions.length > 1) { - setHighlightedOptions([option]) - } else { - setHighlightedOptions( - toggleOption(highlightedOptions, option).slice(-1) - ) - } -} diff --git a/packages/core/src/Transfer/types.js b/packages/core/src/Transfer/types.js deleted file mode 100644 index 55aa9c7e67..0000000000 --- a/packages/core/src/Transfer/types.js +++ /dev/null @@ -1,12 +0,0 @@ -/** - * These are instances of react components, - * both built-in and custom ones. - * - * @typedef {Object} ReactElement - */ - -/** - * @typedef {Object} Option - * @prop {string} label - * @prop {string} value - */ diff --git a/packages/core/src/index.js b/packages/core/src/index.js index a8827e3ca6..4254819aab 100644 --- a/packages/core/src/index.js +++ b/packages/core/src/index.js @@ -69,6 +69,4 @@ export { TableRow } from './Table/TableRow.js' export { TableRowHead } from './Table/TableRowHead.js' export { Tag } from './Tag/Tag.js' export { TextArea } from './TextArea/TextArea.js' -export { Transfer } from './Transfer/Transfer.js' -export { TransferOption } from './Transfer/TransferOption.js' export { Tooltip } from './Tooltip/Tooltip.js' diff --git a/packages/core/src/Transfer/Actions.js b/packages/widgets/src/Transfer/Actions.js similarity index 100% rename from packages/core/src/Transfer/Actions.js rename to packages/widgets/src/Transfer/Actions.js diff --git a/packages/core/src/Transfer/AddAll.js b/packages/widgets/src/Transfer/AddAll.js similarity index 93% rename from packages/core/src/Transfer/AddAll.js rename to packages/widgets/src/Transfer/AddAll.js index b8880e37e4..34ee7a144c 100644 --- a/packages/core/src/Transfer/AddAll.js +++ b/packages/widgets/src/Transfer/AddAll.js @@ -1,7 +1,7 @@ +import { Button } from '@dhis2/ui-core' import React from 'react' import propTypes from '@dhis2/prop-types' -import { Button } from '../Button/Button.js' import { IconAddAll } from './icons.js' export const AddAll = ({ label, dataTest, disabled, onClick }) => ( diff --git a/packages/core/src/Transfer/AddIndividual.js b/packages/widgets/src/Transfer/AddIndividual.js similarity index 93% rename from packages/core/src/Transfer/AddIndividual.js rename to packages/widgets/src/Transfer/AddIndividual.js index acad012d51..9759e8fd8e 100644 --- a/packages/core/src/Transfer/AddIndividual.js +++ b/packages/widgets/src/Transfer/AddIndividual.js @@ -1,7 +1,7 @@ +import { Button } from '@dhis2/ui-core' import React from 'react' import propTypes from '@dhis2/prop-types' -import { Button } from '../Button/Button.js' import { IconAddIndividual } from './icons.js' export const AddIndividual = ({ label, dataTest, disabled, onClick }) => ( diff --git a/packages/core/src/Transfer/Container.js b/packages/widgets/src/Transfer/Container.js similarity index 100% rename from packages/core/src/Transfer/Container.js rename to packages/widgets/src/Transfer/Container.js diff --git a/packages/widgets/src/Transfer/Filter.js b/packages/widgets/src/Transfer/Filter.js new file mode 100644 index 0000000000..00e2c38137 --- /dev/null +++ b/packages/widgets/src/Transfer/Filter.js @@ -0,0 +1,35 @@ +import { Input, Field } from '@dhis2/ui-core' +import { spacers } from '@dhis2/ui-constants' +import React from 'react' +import propTypes from '@dhis2/prop-types' + +export const Filter = ({ dataTest, filter, onChange, label }) => ( +
+ + + + + +
+) + +Filter.propTypes = { + dataTest: propTypes.string.isRequired, + filter: propTypes.string.isRequired, + onChange: propTypes.func.isRequired, + label: propTypes.string, +} diff --git a/packages/core/src/Transfer/LeftFooter.js b/packages/widgets/src/Transfer/LeftFooter.js similarity index 91% rename from packages/core/src/Transfer/LeftFooter.js rename to packages/widgets/src/Transfer/LeftFooter.js index b062679bf0..98fbdf821c 100644 --- a/packages/core/src/Transfer/LeftFooter.js +++ b/packages/widgets/src/Transfer/LeftFooter.js @@ -3,7 +3,7 @@ import propTypes from '@dhis2/prop-types' import { spacers } from '@dhis2/ui-constants' -import { borderColor } from './common.js' +import { borderColor } from './common/index.js' export const LeftFooter = ({ children, dataTest }) => (
diff --git a/packages/core/src/Transfer/LeftHeader.js b/packages/widgets/src/Transfer/LeftHeader.js similarity index 91% rename from packages/core/src/Transfer/LeftHeader.js rename to packages/widgets/src/Transfer/LeftHeader.js index b118ba4337..738959546d 100644 --- a/packages/core/src/Transfer/LeftHeader.js +++ b/packages/widgets/src/Transfer/LeftHeader.js @@ -3,7 +3,7 @@ import propTypes from '@dhis2/prop-types' import { spacers } from '@dhis2/ui-constants' -import { borderColor } from './common.js' +import { borderColor } from './common/index.js' export const LeftHeader = ({ children, dataTest }) => (
diff --git a/packages/core/src/Transfer/LeftSide.js b/packages/widgets/src/Transfer/LeftSide.js similarity index 93% rename from packages/core/src/Transfer/LeftSide.js rename to packages/widgets/src/Transfer/LeftSide.js index 181444ffd2..fcd5d137d2 100644 --- a/packages/core/src/Transfer/LeftSide.js +++ b/packages/widgets/src/Transfer/LeftSide.js @@ -1,7 +1,7 @@ import React from 'react' import propTypes from '@dhis2/prop-types' -import { borderColor, borderRadius } from './common.js' +import { borderColor, borderRadius } from './common/index.js' export const LeftSide = ({ children, dataTest, width }) => (
diff --git a/packages/widgets/src/Transfer/PickedOptions.js b/packages/widgets/src/Transfer/PickedOptions.js new file mode 100644 index 0000000000..78a41a906d --- /dev/null +++ b/packages/widgets/src/Transfer/PickedOptions.js @@ -0,0 +1,29 @@ +import React from 'react' +import propTypes from '@dhis2/prop-types' + +import { spacers } from '@dhis2/ui-constants' + +export const PickedOptions = ({ + children, + dataTest, + selectedEmptyComponent, +}) => ( +
+ {!React.Children.count(children) && selectedEmptyComponent} + {children} + + +
+) + +PickedOptions.propTypes = { + children: propTypes.node.isRequired, + dataTest: propTypes.string.isRequired, + selectedEmptyComponent: propTypes.node, +} diff --git a/packages/core/src/Transfer/RemoveAll.js b/packages/widgets/src/Transfer/RemoveAll.js similarity index 93% rename from packages/core/src/Transfer/RemoveAll.js rename to packages/widgets/src/Transfer/RemoveAll.js index 104b1c4c55..0359b97aa7 100644 --- a/packages/core/src/Transfer/RemoveAll.js +++ b/packages/widgets/src/Transfer/RemoveAll.js @@ -1,7 +1,7 @@ +import { Button } from '@dhis2/ui-core' import React from 'react' import propTypes from '@dhis2/prop-types' -import { Button } from '../Button/Button.js' import { IconRemoveAll } from './icons.js' export const RemoveAll = ({ label, dataTest, disabled, onClick }) => ( diff --git a/packages/core/src/Transfer/RemoveIndividual.js b/packages/widgets/src/Transfer/RemoveIndividual.js similarity index 93% rename from packages/core/src/Transfer/RemoveIndividual.js rename to packages/widgets/src/Transfer/RemoveIndividual.js index df382e6a0c..7e5aa58fe2 100644 --- a/packages/core/src/Transfer/RemoveIndividual.js +++ b/packages/widgets/src/Transfer/RemoveIndividual.js @@ -1,7 +1,7 @@ +import { Button } from '@dhis2/ui-core' import React from 'react' import propTypes from '@dhis2/prop-types' -import { Button } from '../Button/Button.js' import { IconRemoveIndividual } from './icons.js' export const RemoveIndividual = ({ label, dataTest, disabled, onClick }) => ( diff --git a/packages/core/src/Transfer/ReorderingActions.js b/packages/widgets/src/Transfer/ReorderingActions.js similarity index 91% rename from packages/core/src/Transfer/ReorderingActions.js rename to packages/widgets/src/Transfer/ReorderingActions.js index b349c03fdc..a86ccee158 100644 --- a/packages/core/src/Transfer/ReorderingActions.js +++ b/packages/widgets/src/Transfer/ReorderingActions.js @@ -1,9 +1,8 @@ +import { Button } from '@dhis2/ui-core' +import { spacers } from '@dhis2/ui-constants' import React from 'react' import propTypes from '@dhis2/prop-types' -import { spacers } from '@dhis2/ui-constants' - -import { Button } from '../Button/Button.js' import { IconMoveDown, IconMoveUp } from './icons.js' export const ReorderingActions = ({ @@ -34,8 +33,8 @@ export const ReorderingActions = ({ dataTest={`${dataTest}-buttonmoveup`} icon={ } /> diff --git a/packages/core/src/Transfer/RightFooter.js b/packages/widgets/src/Transfer/RightFooter.js similarity index 91% rename from packages/core/src/Transfer/RightFooter.js rename to packages/widgets/src/Transfer/RightFooter.js index 29a03ff865..07cfcad483 100644 --- a/packages/core/src/Transfer/RightFooter.js +++ b/packages/widgets/src/Transfer/RightFooter.js @@ -3,7 +3,7 @@ import propTypes from '@dhis2/prop-types' import { spacers } from '@dhis2/ui-constants' -import { borderColor } from './common.js' +import { borderColor } from './common/index.js' export const RightFooter = ({ children, dataTest }) => (
diff --git a/packages/core/src/Transfer/RightSide.js b/packages/widgets/src/Transfer/RightSide.js similarity index 92% rename from packages/core/src/Transfer/RightSide.js rename to packages/widgets/src/Transfer/RightSide.js index df3ab4e5ef..b435577de2 100644 --- a/packages/core/src/Transfer/RightSide.js +++ b/packages/widgets/src/Transfer/RightSide.js @@ -1,7 +1,7 @@ import React from 'react' import propTypes from '@dhis2/prop-types' -import { borderColor, borderRadius } from './common.js' +import { borderColor, borderRadius } from './common/index.js' export const RightSide = ({ children, dataTest, width }) => (
diff --git a/packages/widgets/src/Transfer/SourceOptions.js b/packages/widgets/src/Transfer/SourceOptions.js new file mode 100644 index 0000000000..bfdd24e7fd --- /dev/null +++ b/packages/widgets/src/Transfer/SourceOptions.js @@ -0,0 +1,29 @@ +import React from 'react' +import propTypes from '@dhis2/prop-types' + +import { spacers } from '@dhis2/ui-constants' + +export const SourceOptions = ({ + children, + dataTest, + sourceEmptyPlaceholder, +}) => ( +
+ {children} + {!React.Children.count(children) && sourceEmptyPlaceholder} + + +
+) + +SourceOptions.propTypes = { + dataTest: propTypes.string.isRequired, + children: propTypes.node, + sourceEmptyPlaceholder: propTypes.node, +} diff --git a/packages/core/src/Transfer/Transfer.js b/packages/widgets/src/Transfer/Transfer.js similarity index 69% rename from packages/core/src/Transfer/Transfer.js rename to packages/widgets/src/Transfer/Transfer.js index 6f2b54c2a5..84d95e7d16 100644 --- a/packages/core/src/Transfer/Transfer.js +++ b/packages/widgets/src/Transfer/Transfer.js @@ -1,6 +1,4 @@ -import './types.js' - -import React, { Children, useState } from 'react' +import React, { useState } from 'react' import propTypes from '@dhis2/prop-types' import { Actions } from './Actions.js' @@ -18,14 +16,13 @@ import { ReorderingActions } from './ReorderingActions.js' import { RightFooter } from './RightFooter.js' import { RightSide } from './RightSide.js' import { SourceOptions } from './SourceOptions.js' +import { TransferOption } from './TransferOption.js' import { addAllSelectableSourceOptions, addIndividualSourceOptions, createDoubleClickHandlers, - extractPickedReactOptions, defaultFilterCallback, - filterOutOptions, - getSubsetByFilter, + getOptionClickHandlers, isReorderDownDisabled, isReorderUpDisabled, moveHighlightedPickedOptionDown, @@ -33,17 +30,9 @@ import { removeAllPickedOptions, removeIndividualPickedOptions, useHighlightedOptions, -} from './helper/index.js' -import { filterReactOptionsBy } from './helper/filterReactOptionsBy.js' - -// TODO: This will be refactored away to match the MultiSelect -export const singleSelectedPropType = propTypes.shape({ - label: propTypes.string, - value: propTypes.string, -}) +} from './Transfer/index.js' -export const multiSelectedPropType = propTypes.arrayOf(singleSelectedPropType) -// ODOT +const identity = value => value /** * @module @@ -75,9 +64,9 @@ export const multiSelectedPropType = propTypes.arrayOf(singleSelectedPropType) * rest of the library */ export const Transfer = ({ + options, onChange, - children, className, dataTest, disabled, @@ -96,6 +85,7 @@ export const Transfer = ({ leftFooter, leftHeader, maxSelections, + optionComponent: TransferOption, optionsWidth, rightFooter, searchTerm, @@ -113,69 +103,78 @@ export const Transfer = ({ */ const [internalFilter, setInternalFilter] = useState(initialSearchTerm) const actualFilter = onFilterChange ? searchTerm : internalFilter + const actualFilterCallback = filterable ? filterCallback : identity /* - * These are all the not-selected option react elements. - * It will replace all selected options with null + * Extract the not-selected options. + * Filters options if filterable is true. */ - const sourceOptions = filterOutOptions(children, selected) - const filteredSourceOptions = getSubsetByFilter({ - reactOptions: sourceOptions, - filter: actualFilter, - filterable, - filterCallback, - }) + const sourceOptions = actualFilterCallback( + options.filter(({ value }) => !selected.includes(value)), + actualFilter + ) /* - * Extract the selected options. This way custom options are supported - * without having to provide a component via the props - * - * Children are sorted by the order given in the "selected" array. - * This is done in order to cover the "append newly selected items - * at the end" feature/behavior. + * Extract the selected options. Can't use `options.filter` + * because we need to keep the order of `selected` */ - const pickedOptions = extractPickedReactOptions({ - reactOptions: children, - selectedPlainOptions: selected, - }) + const pickedOptions = selected + .map(value => options.find(option => value === option.value)) + // filter -> in case a selected value has been provided + // that does not exist as option + .filter(identity) /* - * These are all the highlighted option react elements on the options side. + * These are all the highlighted options on the options side. */ const { highlightedOptions: highlightedSourceOptions, setHighlightedOptions: setHighlightedSourceOptions, toggleHighlightedOption: toggleHighlightedSourceOption, } = useHighlightedOptions({ - reactOptions: filteredSourceOptions, + options: sourceOptions, disabled, maxSelections, }) /* - * These are all the highlighted option react elements on the selected side. + * These are all the highlighted options on the selected side. */ const { highlightedOptions: highlightedPickedOptions, setHighlightedOptions: setHighlightedPickedOptions, toggleHighlightedOption: toggleHighlightedPickedOption, } = useHighlightedOptions({ - reactOptions: pickedOptions, + options: pickedOptions, disabled, maxSelections, }) + /* + * These are the double click handlers for (de-)selection + */ const { selectSingleOption, deselectSingleOption, } = createDoubleClickHandlers({ - selectedPlainOptions: selected, + selected, setHighlightedSourceOptions, setHighlightedPickedOptions, onChange, maxSelections, }) + /** + * Disabled button states + */ + const isAddAllDisabled = + disabled || + sourceOptions.filter(({ disabled }) => !disabled).length === 0 + const isAddIndividualDisabled = disabled || !highlightedSourceOptions.length + const isRemoveAllDisabled = disabled || !selected.length + const isRemoveIndividualDisabled = + disabled || !highlightedPickedOptions.length + return ( @@ -202,13 +201,28 @@ export const Transfer = ({ - {filteredSourceOptions} + {sourceOptions.map(option => { + const Option = option.component || TransferOption + const highlighted = !!highlightedSourceOptions.find( + highlightedSourceOption => + highlightedSourceOption === option.value + ) + + return ( + {leftFooter && ( @@ -223,19 +237,11 @@ export const Transfer = ({ !disabled, - filteredSourceOptions - ) - ) - } + disabled={isAddAllDisabled} onClick={() => addAllSelectableSourceOptions({ - sourceReactOptions: filteredSourceOptions, - selectedPlainOptions: selected, + sourceOptions, + selected, onChange, setHighlightedSourceOptions, }) @@ -246,13 +252,13 @@ export const Transfer = ({ addIndividualSourceOptions({ filterable, - filteredSourcePlainOptions: filteredSourceOptions, - highlightedSourcePlainOptions: highlightedSourceOptions, - selectedPlainOptions: selected, + sourceOptions, + highlightedSourceOptions, + selected, maxSelections, onChange, setHighlightedSourceOptions, @@ -264,7 +270,7 @@ export const Transfer = ({ removeAllPickedOptions({ setHighlightedPickedOptions, @@ -277,12 +283,12 @@ export const Transfer = ({ removeIndividualPickedOptions({ - highlightedPickedReactOptions: highlightedPickedOptions, + highlightedPickedOptions, onChange, - selectedPlainOptions: selected, + selected, setHighlightedPickedOptions, }) } @@ -293,13 +299,27 @@ export const Transfer = ({ - {pickedOptions} + {pickedOptions.map(option => { + const Option = option.component || TransferOption + const highlighted = !!highlightedPickedOptions.find( + value => option.value === value + ) + + return ( + {(rightFooter || enableOrderChange) && ( @@ -308,24 +328,24 @@ export const Transfer = ({ moveHighlightedPickedOptionUp({ - selectedPlainOptions: selected, - highlightedPickedPlainOptions: highlightedPickedOptions, + selected, + highlightedPickedOptions, onChange, }) } onChangeDown={() => { moveHighlightedPickedOptionDown({ - selectedPlainOptions: selected, - highlightedPickedPlainOptions: highlightedPickedOptions, + selected, + highlightedPickedOptions, onChange, }) }} @@ -342,12 +362,13 @@ export const Transfer = ({ Transfer.defaultProps = { dataTest: 'dhis2-uicore-transfer', - initialSearchTerm: '', - selected: [], height: '240px', + initialSearchTerm: '', + maxSelections: Infinity, + optionComponent: TransferOption, optionsWidth: '320px', + selected: [], selectedWidth: '320px', - maxSelections: Infinity, filterCallback: defaultFilterCallback, } @@ -378,16 +399,23 @@ Transfer.defaultProps = { * @prop {string} [removeIndividualText] * @prop {Node} [rightFooter] * @prop {string} [searchTerm] - * @prop {Option|Option[]} selected + * @prop {string[]} selected * @prop {string} [selectedWidth] * @prop {Function} [onFilterChange] */ Transfer.propTypes = { + options: propTypes.arrayOf( + propTypes.shape({ + label: propTypes.string.isRequired, + value: propTypes.string.isRequired, + component: propTypes.func, + disabled: propTypes.bool, + }) + ).isRequired, onChange: propTypes.func.isRequired, addAllText: propTypes.string, addIndividualText: propTypes.string, - children: propTypes.node, className: propTypes.string, dataTest: propTypes.string, disabled: propTypes.bool, @@ -401,15 +429,13 @@ Transfer.propTypes = { leftFooter: propTypes.node, leftHeader: propTypes.node, maxSelections: propTypes.oneOf([1, Infinity]), + optionComponent: propTypes.func, optionsWidth: propTypes.string, removeAllText: propTypes.string, removeIndividualText: propTypes.string, rightFooter: propTypes.node, searchTerm: propTypes.string, - selected: propTypes.oneOfType([ - singleSelectedPropType, - multiSelectedPropType, - ]), + selected: propTypes.arrayOf(propTypes.string), selectedEmptyComponent: propTypes.node, selectedWidth: propTypes.string, sourceEmptyPlaceholder: propTypes.node, diff --git a/packages/core/src/Transfer/Transfer.stories.js b/packages/widgets/src/Transfer/Transfer.stories.js similarity index 51% rename from packages/core/src/Transfer/Transfer.stories.js rename to packages/widgets/src/Transfer/Transfer.stories.js index dc75d78db2..36d4e110dd 100644 --- a/packages/core/src/Transfer/Transfer.stories.js +++ b/packages/widgets/src/Transfer/Transfer.stories.js @@ -1,21 +1,13 @@ /* eslint-disable react/prop-types */ +import { SingleSelectOption, Tab, TabBar } from '@dhis2/ui-core' import React, { useState } from 'react' -import { - Field, - SingleSelect, - SingleSelectOption, - Tab, - TabBar, - Transfer, - TransferOption, -} from '../index.js' +import { SingleSelectField, Transfer } from '../index.js' -export default { title: 'Transfer' } +export default { title: 'Components/widgets/Transfer' } const StatefulWrapper = ({ children, initialState }) => { - const initialSelected = initialState.map(child => child.props) - const [selected, setSelected] = useState(initialSelected) + const [selected, setSelected] = useState(initialState) return React.Children.map(children, child => React.cloneElement(child, { @@ -30,116 +22,96 @@ StatefulWrapper.defaultProps = { } const options = [ - , - , - , - , - , - , - , - , - , - , - , - , - , - , - , - , - , - , - , - , - , - , + { + label: 'ANC 1st visit', + value: 'anc_1st_visit', + }, + { + label: 'ANC 2nd visit', + value: 'anc_2nd_visit', + }, + { + label: 'ANC 3rd visit', + value: 'anc_3rd_visit', + }, + { + label: 'ANC 4th or more visits', + value: 'anc_4th_or_more_visits', + }, + { + label: 'ARI treated with antibiotics (pneumonia) follow-up', + value: 'ari_treated_with_antibiotics_(pneumonia)_follow-up', + }, + { + label: 'ARI treated with antibiotics (pneumonia) new', + value: 'ari_treated_with_antibiotics_(pneumonia)_new', + }, + { + label: 'ARI treated with antibiotics (pneumonia) referrals', + value: 'ari_treated_with_antibiotics_(pneumonia)_referrals', + }, + { + label: 'ARI treated without antibiotics (cough) follow-up', + value: 'ari_treated_without_antibiotics_(cough)_follow-up', + }, + { + label: 'ARI treated without antibiotics (cough) new', + value: 'ari_treated_without_antibiotics_(cough)_new', + }, + { + label: 'ARI treated without antibiotics (cough) referrals', + value: 'ari_treated_without_antibiotics_(cough)_referrals', + }, + { + label: 'ART No clients who stopped TRT due to TRT failure', + value: 'art_no_clients_who_stopped_trt_due_to_trt_failure', + }, + { + label: + 'ART No clients who stopped TRT due to adverse clinical status/event', + value: + 'art_no_clients_who_stopped_trt_due_to_adverse_clinical_status/event', + }, + { + label: 'ART No clients with change of regimen due to drug toxicity', + value: 'art_no_clients_with_change_of_regimen_due_to_drug_toxicity', + }, + { + label: 'ART No clients with new adverse drug reaction', + value: 'art_no_clients_with_new_adverse_drug_reaction', + }, + { + label: 'ART No started Opportunist Infection prophylaxis', + value: 'art_no_started_opportunist_infection_prophylaxis', + }, + { + label: 'ART clients with new adverse clinical event', + value: 'art_clients_with_new_adverse_clinical_event', + }, + { + label: 'ART defaulters', + value: 'art_defaulters', + }, + { + label: 'ART enrollment stage 1', + value: 'art_enrollment_stage_1', + }, + { + label: 'ART enrollment stage 2', + value: 'art_enrollment_stage_2', + }, + { + label: 'ART enrollment stage 3', + value: 'art_enrollment_stage_3', + }, + { + label: 'ART enrollment stage 4', + value: 'art_enrollment_stage_4', + }, + { + label: 'ART entry point: No PMTCT', + value: 'art_entry_point:_no_pmtct', + }, ] export const SingleSelection = () => ( @@ -147,17 +119,17 @@ export const SingleSelection = () => ( console.log('Will be overriden')} - > - {options} - + options={options} + /> ) export const Multiple = () => ( - console.log('Will be overriden')}> - {options} - + console.log('Will be overriden')} + options={options.slice(0, 3)} + /> ) @@ -166,9 +138,8 @@ export const Header = () => ( console.log('Will be overriden')} leftHeader={

Header on the left side

} - > - {options} -
+ options={options} + /> ) @@ -188,9 +159,8 @@ export const OptionsFooter = () => ( Reload list } - > - {options} - + options={options} + /> ) @@ -201,9 +171,8 @@ export const Filtered = () => ( onChange={() => console.log('Will be overriden by StatefulWrapper')} initialSearchTerm="ANC" leftHeader={

Header on the left side

} - > - {options} - + options={options} + /> ) @@ -234,19 +203,29 @@ export const CustomListOptions = () => ( onChange={() => console.log('Will be overriden by StatefulWrapper') } - optionsComponent={CustomOption} - > - - - - - - - + optionComponent={CustomOption} + options={options} + /> ) +export const IndividualCustomOption = () => ( + + console.log('Will be overriden')} + addAllText="Add all" + addIndividualText="Add individual" + removeAllText="Remove all" + removeIndividualText="Remove individual" + options={[ + { ...options[0], component: CustomOption }, + ...options.slice(1), + ]} + /> + +) + export const CustomButtonText = () => ( ( addIndividualText="Add individual" removeAllText="Remove all" removeIndividualText="Remove individual" - > - {options} - + options={options} + /> ) export const SourceEmptyPlaceholder = () => ( console.log('Will be overriden')} + options={[]} sourceEmptyPlaceholder={

No options found. @@ -285,16 +264,19 @@ export const PickedEmptyComponent = () => (

} - > - {options} -
+ options={options} + /> ) export const Reordering = () => ( - - null}> - {options.slice(0, 4)} - + value)} + > + null} + options={options.slice(0, 4)} + /> ) @@ -309,9 +291,8 @@ export const IncreasedOptionsHeight = () => ( } height="400px" leftHeader={

Header on the left side

} - > - {options} - + options={options} + />
) @@ -325,34 +306,23 @@ export const DifferentWidths = () => ( leftHeader={

Header on the left side

} optionsWidth="500px" selectedWidth="240px" - > - {options} - + options={options} + /> ) const createCustomFilteringInHeader = hideFilterInput => { - const relativePeriods = React.Children.map( - options.slice(0, 10), - (child, index) => - React.cloneElement(child, { - additionalData: { - relativePeriod: true, - year: index < 5 ? '2020' : '2019', - }, - }) - ) - - const fixedPeriods = React.Children.map( - options.slice(10, 20), - (child, index) => - React.cloneElement(child, { - additionalData: { - relativePeriod: false, - year: index < 5 ? '2020' : '2019', - }, - }) - ) + const relativePeriods = options.slice(0, 10).map((option, index) => ({ + ...option, + relativePeriod: true, + year: index < 5 ? '2020' : '2019', + })) + + const fixedPeriods = options.slice(10, 20).map((option, index) => ({ + ...option, + relativePeriod: false, + year: index < 5 ? '2020' : '2019', + })) const allOptions = [...relativePeriods, ...fixedPeriods] @@ -381,25 +351,24 @@ const createCustomFilteringInHeader = hideFilterInput => {

- - - - - - + + + + ) const CustomTransfer = props => { const [filter, setFilter] = useState('') const [relativePeriod, setRelativePeriod] = useState(true) - const [year, setYear] = useState({ label: '2020', value: '2020' }) + const [year, setYear] = useState('2020') const filterCallback = (options, filter) => { const optionsWithYear = options.filter( - option => option.year === year.value + option => option.year === year ) const optionsWithPeriod = optionsWithYear.filter( @@ -442,7 +411,7 @@ const createCustomFilteringInHeader = hideFilterInput => { // eslint-disable-next-line react/display-name return () => ( - {allOptions} + ) } diff --git a/packages/core/src/Transfer/helper/addAllSelectableSourceOptions.js b/packages/widgets/src/Transfer/Transfer/addAllSelectableSourceOptions.js similarity index 54% rename from packages/core/src/Transfer/helper/addAllSelectableSourceOptions.js rename to packages/widgets/src/Transfer/Transfer/addAllSelectableSourceOptions.js index a5a23d495a..55735e3517 100644 --- a/packages/core/src/Transfer/helper/addAllSelectableSourceOptions.js +++ b/packages/widgets/src/Transfer/Transfer/addAllSelectableSourceOptions.js @@ -1,26 +1,27 @@ -import { addOption, toggleOptions } from '../common' -import { getPlainOptionsFromReactOptions } from './getPlainOptionsFromReactOptions' - /** * @param {Object} args - * @param {ReactElement} args.sourceReactOptions - * @param {Option[]} args.selectedPlainOptions + * @param {Object[]} args.sourceOptions + * @param {string[]} args.selected * @param {Function} args.onChange * @param {Function} arg.setHighlightedSourceOptions * @returns {void} */ export const addAllSelectableSourceOptions = ({ - sourceReactOptions, + sourceOptions, onChange, - selectedPlainOptions, + selected, setHighlightedSourceOptions, }) => { - const all = getPlainOptionsFromReactOptions(sourceReactOptions) - const allEnabled = all.filter(({ disabled }) => !disabled) - const newSelected = toggleOptions( - selectedPlainOptions, - allEnabled, - addOption + const enabledSourceOptions = sourceOptions.filter( + ({ disabled }) => !disabled + ) + + const newSelected = enabledSourceOptions.reduce( + (accumulatedSelected, enabledSourceOption) => [ + ...accumulatedSelected, + enabledSourceOption.value, + ], + selected ) setHighlightedSourceOptions([]) diff --git a/packages/core/src/Transfer/helper/addIndividualSourceOptions.js b/packages/widgets/src/Transfer/Transfer/addIndividualSourceOptions.js similarity index 60% rename from packages/core/src/Transfer/helper/addIndividualSourceOptions.js rename to packages/widgets/src/Transfer/Transfer/addIndividualSourceOptions.js index 404a38c9f3..52f187298d 100644 --- a/packages/core/src/Transfer/helper/addIndividualSourceOptions.js +++ b/packages/widgets/src/Transfer/Transfer/addIndividualSourceOptions.js @@ -1,23 +1,20 @@ -import { Children } from 'react' -import { addOption, isOption, toggleOptions } from '../common' - /** * @param {Object} args * @param {bool} args.filterable - * @param {ReactElement} args.filteredSourcePlainOptions - * @param {Option[]} args.highlightedSourcePlainOptions + * @param {Object[]} args.sourceOptions + * @param {string[]} args.highlightedSourceOptions + * @param {string[]} args.selected * @param {Function} args.onChange - * @param {Option[]} args.selectedPlainOptions * @param {Function} args.setHighlightedSourceOptions * @returns void */ export const addIndividualSourceOptions = ({ filterable, - filteredSourcePlainOptions, - highlightedSourcePlainOptions, + sourceOptions, + highlightedSourceOptions, maxSelections, onChange, - selectedPlainOptions, + selected, setHighlightedSourceOptions, }) => { /** @@ -32,17 +29,22 @@ export const addIndividualSourceOptions = ({ * clicks the "add individuals" button */ const filteredHighlightedSourceOptions = filterable - ? highlightedSourcePlainOptions.filter(option => - Children.toArray(filteredSourcePlainOptions) - .map(({ props }) => props) - .find(filteredOption => isOption(filteredOption, option)) + ? highlightedSourceOptions.filter(value => + sourceOptions.find( + filteredOption => filteredOption.value === value + ) ) - : highlightedSourcePlainOptions + : highlightedSourceOptions - const newSelected = toggleOptions( - selectedPlainOptions, - filteredHighlightedSourceOptions, - addOption + const newSelected = filteredHighlightedSourceOptions.reduce( + (accumulatedSelected, value) => [ + ...accumulatedSelected, + filteredHighlightedSourceOptions.find( + filteredHighlightedSourceOption => + filteredHighlightedSourceOption === value + ), + ], + selected ) setHighlightedSourceOptions([]) diff --git a/packages/core/src/Transfer/helper/createDoubleClickHandlers.js b/packages/widgets/src/Transfer/Transfer/createDoubleClickHandlers.js similarity index 72% rename from packages/core/src/Transfer/helper/createDoubleClickHandlers.js rename to packages/widgets/src/Transfer/Transfer/createDoubleClickHandlers.js index 37637e6e33..f1a3595f06 100644 --- a/packages/core/src/Transfer/helper/createDoubleClickHandlers.js +++ b/packages/widgets/src/Transfer/Transfer/createDoubleClickHandlers.js @@ -1,10 +1,8 @@ -import { addOption, removeOption } from '../common' - /** * @param {Object} args * @param {number} args.maxSelections + * @param {string[]} args.selected * @param {Function} args.onChange - * @param {Option[]} args.selectedPlainOptions * @param {Function} args.setHighlightedSourceOptions * @param {Function} args.setHighlightedPickedOptions * @returns void @@ -12,18 +10,24 @@ import { addOption, removeOption } from '../common' export const createDoubleClickHandlers = ({ maxSelections, onChange, - selectedPlainOptions, + selected, setHighlightedPickedOptions, setHighlightedSourceOptions, }) => { const selectSingleOption = ({ option }) => { - const newSelected = addOption(selectedPlainOptions, option) + const newSelected = selected.includes(option.value) + ? selected + : [...selected, option.value] + setHighlightedSourceOptions([]) onChange({ selected: newSelected.slice(-1 * maxSelections) }) } const deselectSingleOption = ({ option }) => { - const newSelected = removeOption(selectedPlainOptions, option) + const newSelected = selected.filter( + curSelected => curSelected !== option.value + ) + setHighlightedPickedOptions([]) onChange({ selected: newSelected }) } diff --git a/packages/widgets/src/Transfer/Transfer/defaultFilterCallback.js b/packages/widgets/src/Transfer/Transfer/defaultFilterCallback.js new file mode 100644 index 0000000000..c895f141d2 --- /dev/null +++ b/packages/widgets/src/Transfer/Transfer/defaultFilterCallback.js @@ -0,0 +1,9 @@ +/** + * @param {Object[]} options + * @param {string} filter + * @returns {Object[]} + */ +export const defaultFilterCallback = (options, filter) => + filter === '' + ? options + : options.filter(({ label }) => label.match(new RegExp(filter, 'i'))) diff --git a/packages/widgets/src/Transfer/Transfer/getOptionClickHandlers.js b/packages/widgets/src/Transfer/Transfer/getOptionClickHandlers.js new file mode 100644 index 0000000000..5ee5de09a1 --- /dev/null +++ b/packages/widgets/src/Transfer/Transfer/getOptionClickHandlers.js @@ -0,0 +1,19 @@ +import { getModeByModifierKey } from '../common/index.js' + +/** + * @param {Object} option + * @param {Function} selectionHandler + * @param {Function} toggleHighlightedOption + * @returns {Object} + */ +export const getOptionClickHandlers = ( + option, + selectionHandler, + toggleHighlightedOption +) => ({ + onClick: (_, event) => { + const mode = getModeByModifierKey(event) + toggleHighlightedOption({ option, mode }) + }, + onDoubleClick: selectionHandler, +}) diff --git a/packages/core/src/Transfer/helper/index.js b/packages/widgets/src/Transfer/Transfer/index.js similarity index 74% rename from packages/core/src/Transfer/helper/index.js rename to packages/widgets/src/Transfer/Transfer/index.js index 1f8f66c186..8e7e3662f8 100644 --- a/packages/core/src/Transfer/helper/index.js +++ b/packages/widgets/src/Transfer/Transfer/index.js @@ -1,11 +1,8 @@ export * from './addAllSelectableSourceOptions.js' export * from './addIndividualSourceOptions.js' export * from './createDoubleClickHandlers.js' -export * from './extractPickedReactOptions.js' export * from './defaultFilterCallback.js' -export * from './filterOutOptions.js' -export * from './getPlainOptionsFromReactOptions.js' -export * from './getSubsetByFilter.js' +export * from './getOptionClickHandlers.js' export * from './isReorderDownDisabled.js' export * from './isReorderUpDisabled.js' export * from './moveHighlightedPickedOptionDown.js' diff --git a/packages/widgets/src/Transfer/Transfer/isReorderDownDisabled.js b/packages/widgets/src/Transfer/Transfer/isReorderDownDisabled.js new file mode 100644 index 0000000000..10f1042db6 --- /dev/null +++ b/packages/widgets/src/Transfer/Transfer/isReorderDownDisabled.js @@ -0,0 +1,11 @@ +/** + * @param {Object} args + * @param {string} args.highlightedPickedOptions + * @param {string[]} args.selected + * @returns {bool} + */ +export const isReorderDownDisabled = ({ highlightedPickedOptions, selected }) => + // only one item can be moved with the buttons + highlightedPickedOptions.length !== 1 || + // can't move an item down if it's the last one + selected.indexOf(highlightedPickedOptions[0]) === selected.length - 1 diff --git a/packages/widgets/src/Transfer/Transfer/isReorderUpDisabled.js b/packages/widgets/src/Transfer/Transfer/isReorderUpDisabled.js new file mode 100644 index 0000000000..9386ac169e --- /dev/null +++ b/packages/widgets/src/Transfer/Transfer/isReorderUpDisabled.js @@ -0,0 +1,11 @@ +/** + * @param {Object} args + * @param {string} args.highlightedPickedOptions + * @param {string[]} args.selected + * @returns {bool} + */ +export const isReorderUpDisabled = ({ highlightedPickedOptions, selected }) => + // only one item can be moved with the buttons + highlightedPickedOptions.length !== 1 || + // can't move an item up if it's the first one + selected.indexOf(highlightedPickedOptions[0]) === 0 diff --git a/packages/widgets/src/Transfer/Transfer/moveHighlightedPickedOptionDown.js b/packages/widgets/src/Transfer/Transfer/moveHighlightedPickedOptionDown.js new file mode 100644 index 0000000000..eb015b392e --- /dev/null +++ b/packages/widgets/src/Transfer/Transfer/moveHighlightedPickedOptionDown.js @@ -0,0 +1,29 @@ +/** + * @param {Object} args + * @param {string[]} args.selected + * @param {string[]} args.highlightedPickedOptions + * @param {Function} args.onChange + * @returns {void} + */ +export const moveHighlightedPickedOptionDown = ({ + selected, + highlightedPickedOptions, + onChange, +}) => { + const optionIndex = selected.findIndex( + selectedOption => selectedOption === highlightedPickedOptions[0] + ) + + // Can't move down last or non-existing option + if (optionIndex === -1 || optionIndex > selected.length - 2) return + + // swap with next item + const reordered = [ + ...selected.slice(0, optionIndex), + selected[optionIndex + 1], + selected[optionIndex], + ...selected.slice(optionIndex + 2), + ] + + onChange({ selected: reordered }) +} diff --git a/packages/widgets/src/Transfer/Transfer/moveHighlightedPickedOptionUp.js b/packages/widgets/src/Transfer/Transfer/moveHighlightedPickedOptionUp.js new file mode 100644 index 0000000000..44611779ca --- /dev/null +++ b/packages/widgets/src/Transfer/Transfer/moveHighlightedPickedOptionUp.js @@ -0,0 +1,29 @@ +/** + * @param {Object} args + * @param {string[]} args.selected + * @param {string[]} args.highlightedPickedOptions + * @param {Function} args.onChange + * @returns {void} + */ +export const moveHighlightedPickedOptionUp = ({ + selected, + highlightedPickedOptions, + onChange, +}) => { + const optionIndex = selected.findIndex( + selectedOption => selectedOption === highlightedPickedOptions[0] + ) + + // Can't move up option at index 0 or non-existing option + if (optionIndex < 1) return + + // swap with previous item + const reordered = [ + ...selected.slice(0, optionIndex - 1), + selected[optionIndex], + selected[optionIndex - 1], + ...selected.slice(optionIndex + 1), + ] + + onChange({ selected: reordered }) +} diff --git a/packages/core/src/Transfer/helper/removeAllPickedOptions.js b/packages/widgets/src/Transfer/Transfer/removeAllPickedOptions.js similarity index 100% rename from packages/core/src/Transfer/helper/removeAllPickedOptions.js rename to packages/widgets/src/Transfer/Transfer/removeAllPickedOptions.js diff --git a/packages/core/src/Transfer/helper/removeIndividualPickedOptions.js b/packages/widgets/src/Transfer/Transfer/removeIndividualPickedOptions.js similarity index 58% rename from packages/core/src/Transfer/helper/removeIndividualPickedOptions.js rename to packages/widgets/src/Transfer/Transfer/removeIndividualPickedOptions.js index 90b6a022cc..d5462487b8 100644 --- a/packages/core/src/Transfer/helper/removeIndividualPickedOptions.js +++ b/packages/widgets/src/Transfer/Transfer/removeIndividualPickedOptions.js @@ -1,23 +1,19 @@ -import { removeOption, toggleOptions } from '../common' - /** * @param {Object} args - * @param {ReactElement} args.highlightedPickedReactOptions + * @param {string[]} args.highlightedPickedOptions + * @param {string[]} args.selected * @param {Function} args.setHighlightedPickedOptions - * @param {Option[]} args.selectedPlainOptions * @param {Function} args.onChange * @returns {void} */ export const removeIndividualPickedOptions = ({ - highlightedPickedReactOptions, + highlightedPickedOptions, onChange, - selectedPlainOptions, + selected, setHighlightedPickedOptions, }) => { - const newSelected = toggleOptions( - selectedPlainOptions, - highlightedPickedReactOptions, - removeOption + const newSelected = selected.filter( + selectedOption => !highlightedPickedOptions.includes(selectedOption) ) setHighlightedPickedOptions([]) diff --git a/packages/core/src/Transfer/helper/useHighlightedOptions.js b/packages/widgets/src/Transfer/Transfer/useHighlightedOptions.js similarity index 79% rename from packages/core/src/Transfer/helper/useHighlightedOptions.js rename to packages/widgets/src/Transfer/Transfer/useHighlightedOptions.js index d4a3d18aae..90175adc1a 100644 --- a/packages/core/src/Transfer/helper/useHighlightedOptions.js +++ b/packages/widgets/src/Transfer/Transfer/useHighlightedOptions.js @@ -1,21 +1,15 @@ -import '../types.js' - import { useState } from 'react' -import { createToggleHighlightedOption } from './useHighlightedOptions/createToggleHighlightedOption' +import { createToggleHighlightedOption } from './useHighlightedOptions/createToggleHighlightedOption.js' /** * @param {Object} args * @param {bool} args.disabled * @param {number} args.maxSelection - * @param {ReactElement} args.reactOptions + * @param {Object[]} args.options * @returns {Object} highlighted options & helpers */ -export const useHighlightedOptions = ({ - disabled, - maxSelections, - reactOptions, -}) => { +export const useHighlightedOptions = ({ disabled, maxSelections, options }) => { /** * These are important so the stored element can be used * as range-start when using shift multiple times consecutively @@ -29,7 +23,7 @@ export const useHighlightedOptions = ({ setHighlightedOptions, maxSelections, setLastClicked, - reactOptions, + options, lastClicked, }) diff --git a/packages/core/src/Transfer/helper/useHighlightedOptions/createToggleHighlightedOption.js b/packages/widgets/src/Transfer/Transfer/useHighlightedOptions/createToggleHighlightedOption.js similarity index 55% rename from packages/core/src/Transfer/helper/useHighlightedOptions/createToggleHighlightedOption.js rename to packages/widgets/src/Transfer/Transfer/useHighlightedOptions/createToggleHighlightedOption.js index 153eeae2b7..fc2ceba8e3 100644 --- a/packages/core/src/Transfer/helper/useHighlightedOptions/createToggleHighlightedOption.js +++ b/packages/widgets/src/Transfer/Transfer/useHighlightedOptions/createToggleHighlightedOption.js @@ -1,15 +1,26 @@ -import { ADD_MODE, RANGE_MODE } from '../../common' -import { toggleAdd } from './toggleAdd' -import { toggleRange } from './toggleRange' -import { toggleReplace } from './toggleReplace' +import { ADD_MODE, RANGE_MODE } from '../../common/index.js' +import { toggleAdd } from './toggleAdd.js' +import { toggleRange } from './toggleRange.js' +import { toggleReplace } from './toggleReplace.js' +/** + * @param {Object} args + * @param {bool} args.disabled + * @param {string[]} args.highlightedOptions + * @param {Function} args.setHighlightedOptions + * @param {number} args.maxSelections + * @param {Function} args.setLastClicked + * @param {Object[]} args.options + * @param {string} args.lastClicked + * @returns {void} + */ export const createToggleHighlightedOption = ({ disabled, highlightedOptions, setHighlightedOptions, maxSelections, setLastClicked, - reactOptions, + options, lastClicked, }) => ({ option, mode }) => { if (disabled) return @@ -17,7 +28,7 @@ export const createToggleHighlightedOption = ({ setHighlightedOptions([]) if (mode === ADD_MODE) { - setLastClicked(option) + setLastClicked(option.value) return toggleAdd({ highlightedOptions, @@ -30,7 +41,7 @@ export const createToggleHighlightedOption = ({ if (mode === RANGE_MODE) { return toggleRange({ highlightedOptions, - reactOptions, + options, option, setHighlightedOptions, lastClicked, @@ -39,7 +50,7 @@ export const createToggleHighlightedOption = ({ } // REPLACE_MODE - setLastClicked(option) + setLastClicked(option.value) return toggleReplace({ option, diff --git a/packages/widgets/src/Transfer/Transfer/useHighlightedOptions/toggleAdd.js b/packages/widgets/src/Transfer/Transfer/useHighlightedOptions/toggleAdd.js new file mode 100644 index 0000000000..30e7f086c6 --- /dev/null +++ b/packages/widgets/src/Transfer/Transfer/useHighlightedOptions/toggleAdd.js @@ -0,0 +1,20 @@ +import { toggleValue } from '../../common' + +/** + * @param {Object} args + * @param {number} args.maxSelections + * @param {string[]} args.highlightedOptions + * @param {Object} args.option + * @param {Function} args.setHighlightedOption + * @returns {void} + */ +export const toggleAdd = ({ + highlightedOptions, + maxSelections, + option, + setHighlightedOptions, +}) => { + const afterToggled = toggleValue(highlightedOptions, option.value) + const capped = afterToggled.slice(-1 * maxSelections) + setHighlightedOptions(capped) +} diff --git a/packages/core/src/Transfer/helper/useHighlightedOptions/toggleRange.js b/packages/widgets/src/Transfer/Transfer/useHighlightedOptions/toggleRange.js similarity index 72% rename from packages/core/src/Transfer/helper/useHighlightedOptions/toggleRange.js rename to packages/widgets/src/Transfer/Transfer/useHighlightedOptions/toggleRange.js index 0db1b82fb6..415a2ac4e6 100644 --- a/packages/core/src/Transfer/helper/useHighlightedOptions/toggleRange.js +++ b/packages/widgets/src/Transfer/Transfer/useHighlightedOptions/toggleRange.js @@ -1,34 +1,31 @@ -import '../../types.js' -import { findOption, findOptionIndex } from '../../common' -import { getPlainOptionsFromReactOptions } from '../getPlainOptionsFromReactOptions' +import { findOptionIndex } from '../../common/index.js' /** * @param {Object} args - * @param {Option[]} args.highlightedOptions - * @param {ReactElement} args.reactOptions - * @param {Option} args.option - * @param {Function} args.setHighlightedOption * @param {number} args.maxSelections - * @param {Option} args.lastClicked + * @param {string[]} args.highlightedOptions + * @param {Object[]} args.options + * @param {Object} args.option + * @param {string} args.lastClicked + * @param {Function} args.setHighlightedOption * @returns {void} */ export const toggleRange = ({ highlightedOptions, - reactOptions, + options, option, setHighlightedOptions, lastClicked, maxSelections, }) => { if (highlightedOptions.length === 0) { - setHighlightedOptions([option]) + setHighlightedOptions([option.value]) } else { let from, to - const options = getPlainOptionsFromReactOptions(reactOptions) const clickedOptionIndex = findOptionIndex(options, option) const lastClickedSourceOptionWithoutRangeModeIndex = lastClicked - ? findOptionIndex(options, lastClicked) + ? options.findIndex(curOption => curOption.value === lastClicked) : -1 if (lastClickedSourceOptionWithoutRangeModeIndex !== -1) { @@ -39,7 +36,9 @@ export const toggleRange = ({ * A filter-change has removed the most recently highlighted option */ const firstHighlightedInList = options.findIndex(option => - findOption(highlightedOptions, option) + highlightedOptions.find( + highlightedOption => highlightedOption === option.value + ) ) from = firstHighlightedInList @@ -55,6 +54,7 @@ export const toggleRange = ({ .slice(lower, higher + 1) .filter(option => !option.disabled) .slice(maxSelections * -1) + .map(({ value }) => value) setHighlightedOptions(newHighlightedSourceOptions) } diff --git a/packages/widgets/src/Transfer/Transfer/useHighlightedOptions/toggleReplace.js b/packages/widgets/src/Transfer/Transfer/useHighlightedOptions/toggleReplace.js new file mode 100644 index 0000000000..9e676e1119 --- /dev/null +++ b/packages/widgets/src/Transfer/Transfer/useHighlightedOptions/toggleReplace.js @@ -0,0 +1,26 @@ +/** + * @param {Object} args + * @param {string[]} args.highlightedOptions + * @param {Object} args.option + * @param {Function} args.setHighlightedOption + * @returns {void} + */ +export const toggleReplace = ({ + option, + highlightedOptions, + setHighlightedOptions, +}) => { + if (highlightedOptions.length > 1) { + setHighlightedOptions([option.value]) + } else { + const optionIndex = highlightedOptions.findIndex( + highlightedOption => highlightedOption === option.value + ) + + if (optionIndex === -1) { + setHighlightedOptions([option.value]) + } else { + setHighlightedOptions([]) + } + } +} diff --git a/packages/core/src/Transfer/TransferOption.js b/packages/widgets/src/Transfer/TransferOption.js similarity index 90% rename from packages/core/src/Transfer/TransferOption.js rename to packages/widgets/src/Transfer/TransferOption.js index dcb4379fda..1b914cfbae 100644 --- a/packages/core/src/Transfer/TransferOption.js +++ b/packages/widgets/src/Transfer/TransferOption.js @@ -20,12 +20,10 @@ export const TransferOption = ({ className, disabled, dataTest, - label, highlighted, onClick, onDoubleClick, - value, - additionalData, + option, }) => { const doubleClickTimeout = useRef(null) @@ -35,8 +33,6 @@ export const TransferOption = ({ onClick={event => { if (disabled) return - const option = { label, value, ...additionalData } - if (doubleClickTimeout.current) { clearTimeout(doubleClickTimeout.current) doubleClickTimeout.current = null @@ -51,10 +47,10 @@ export const TransferOption = ({ onClick({ option }, event) } }} - data-value={value} + data-value={option.value} className={cx(className, { highlighted, disabled })} > - {label} + {option.label}