@commercetools-frontend-extensions/export-resources-modal 4.0.4 → 4.0.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/commercetools-frontend-extensions-export-resources-modal.cjs.dev.js +74 -21
- package/dist/commercetools-frontend-extensions-export-resources-modal.cjs.prod.js +71 -18
- package/dist/commercetools-frontend-extensions-export-resources-modal.esm.js +74 -21
- package/dist/declarations/src/@constants/misc.d.ts +1 -0
- package/package.json +6 -5
|
@@ -21,6 +21,7 @@ import _possibleConstructorReturn from '@babel/runtime-corejs3/helpers/esm/possi
|
|
|
21
21
|
import _getPrototypeOf from '@babel/runtime-corejs3/helpers/esm/getPrototypeOf';
|
|
22
22
|
import _wrapNativeSuper from '@babel/runtime-corejs3/helpers/esm/wrapNativeSuper';
|
|
23
23
|
import React, { useContext, createContext, useState, useEffect } from 'react';
|
|
24
|
+
import _slicedToArray from '@babel/runtime-corejs3/helpers/esm/slicedToArray';
|
|
24
25
|
import _concatInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/concat';
|
|
25
26
|
import moment from 'moment';
|
|
26
27
|
import _someInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/some';
|
|
@@ -31,10 +32,10 @@ import _Array$isArray from '@babel/runtime-corejs3/core-js-stable/array/is-array
|
|
|
31
32
|
import _Set from '@babel/runtime-corejs3/core-js-stable/set';
|
|
32
33
|
import _sortInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/sort';
|
|
33
34
|
import _indexOfInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/index-of';
|
|
35
|
+
import { useStorage } from '@commercetools-frontend/experimental-components';
|
|
34
36
|
import { actions, useAsyncDispatch } from '@commercetools-frontend/sdk';
|
|
35
37
|
import { useShowNotification } from '@commercetools-frontend/actions-global';
|
|
36
38
|
import { MC_API_PROXY_TARGETS, DOMAINS } from '@commercetools-frontend/constants';
|
|
37
|
-
import _slicedToArray from '@babel/runtime-corejs3/helpers/esm/slicedToArray';
|
|
38
39
|
import _styled from '@emotion/styled/base';
|
|
39
40
|
import _everyInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/every';
|
|
40
41
|
import _includesInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/includes';
|
|
@@ -234,6 +235,7 @@ function renderLocaleError(key) {
|
|
|
234
235
|
}
|
|
235
236
|
|
|
236
237
|
const Z_INDEX_DROPDOWN = 30000;
|
|
238
|
+
const OPERATIONS_EXPORT_RESOURCES_MODAL_SELECTION = 'OPERATIONS_EXPORT_RESOURCES_MODAL_SELECTION';
|
|
237
239
|
|
|
238
240
|
const EXPORT_OPERATION_STATES = {
|
|
239
241
|
PROCESSING: 'processing',
|
|
@@ -476,14 +478,27 @@ function tmpBuildFilters(selectedResourceIds) {
|
|
|
476
478
|
}
|
|
477
479
|
|
|
478
480
|
function ownKeys$3(e, r) { var t = _Object$keys(e); if (_Object$getOwnPropertySymbols) { var o = _Object$getOwnPropertySymbols(e); r && (o = _filterInstanceProperty(o).call(o, function (r) { return _Object$getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
|
|
479
|
-
function _objectSpread$3(e) { for (var r = 1; r < arguments.length; r++) { var
|
|
480
|
-
const updateFieldGroupsWithIsCheckedValue = groups => {
|
|
481
|
-
const updateFields = (fields, parentChecked) => {
|
|
481
|
+
function _objectSpread$3(e) { for (var r = 1; r < arguments.length; r++) { var _context4, _context5; var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? _forEachInstanceProperty(_context4 = ownKeys$3(Object(t), !0)).call(_context4, function (r) { _defineProperty(e, r, t[r]); }) : _Object$getOwnPropertyDescriptors ? _Object$defineProperties(e, _Object$getOwnPropertyDescriptors(t)) : _forEachInstanceProperty(_context5 = ownKeys$3(Object(t))).call(_context5, function (r) { _Object$defineProperty(e, r, _Object$getOwnPropertyDescriptor(t, r)); }); } return e; }
|
|
482
|
+
const updateFieldGroupsWithIsCheckedValue = (groups, cachedSelectedGroups) => {
|
|
483
|
+
const updateFields = (fields, parentChecked, fieldName) => {
|
|
482
484
|
return _mapInstanceProperty(fields).call(fields, field => {
|
|
483
|
-
|
|
484
|
-
|
|
485
|
+
let isChecked = false;
|
|
486
|
+
if (field.isRequired) {
|
|
487
|
+
isChecked = true;
|
|
488
|
+
} else {
|
|
489
|
+
var _context;
|
|
490
|
+
if (cachedSelectedGroups[_concatInstanceProperty(_context = "".concat(fieldName, ".")).call(_context, field.name)] !== undefined) {
|
|
491
|
+
var _context2;
|
|
492
|
+
isChecked = cachedSelectedGroups[_concatInstanceProperty(_context2 = "".concat(fieldName, ".")).call(_context2, field.name)];
|
|
493
|
+
} else if (!field.isReadOnly && field.isSelectedByDefault) {
|
|
494
|
+
isChecked = true;
|
|
495
|
+
} else if (parentChecked) {
|
|
496
|
+
isChecked = true;
|
|
497
|
+
}
|
|
498
|
+
}
|
|
485
499
|
if (field.fields) {
|
|
486
|
-
|
|
500
|
+
var _context3;
|
|
501
|
+
const updatedFields = updateFields(field.fields, isChecked, _concatInstanceProperty(_context3 = "".concat(fieldName, ".")).call(_context3, field.name));
|
|
487
502
|
isChecked = false;
|
|
488
503
|
field.fields = updatedFields;
|
|
489
504
|
}
|
|
@@ -493,7 +508,7 @@ const updateFieldGroupsWithIsCheckedValue = groups => {
|
|
|
493
508
|
});
|
|
494
509
|
};
|
|
495
510
|
return _mapInstanceProperty(groups).call(groups, group => {
|
|
496
|
-
const updatedFields = updateFields(group.fields, false);
|
|
511
|
+
const updatedFields = updateFields(group.fields, false, group.groupLabel);
|
|
497
512
|
return _objectSpread$3(_objectSpread$3({}, group), {}, {
|
|
498
513
|
fields: updatedFields
|
|
499
514
|
});
|
|
@@ -519,25 +534,36 @@ const updateFieldGroupWithAdditionalFieldExtensions = groups => {
|
|
|
519
534
|
};
|
|
520
535
|
|
|
521
536
|
const useInitialValues = props => {
|
|
537
|
+
var _context;
|
|
522
538
|
const todayFormatted = moment().format('DD-MM-YY_HH-mm');
|
|
523
539
|
const _useApplicationContex = useApplicationContext(applicationContext => ({
|
|
524
540
|
locale: applicationContext.dataLocale
|
|
525
541
|
})),
|
|
526
542
|
locale = _useApplicationContex.locale;
|
|
527
|
-
const
|
|
543
|
+
const _useApplicationContex2 = useApplicationContext(context => {
|
|
544
|
+
var _context$project;
|
|
545
|
+
return {
|
|
546
|
+
projectKey: (_context$project = context.project) === null || _context$project === void 0 ? void 0 : _context$project.key
|
|
547
|
+
};
|
|
548
|
+
}),
|
|
549
|
+
projectKey = _useApplicationContex2.projectKey;
|
|
550
|
+
const _useStorage = useStorage(_concatInstanceProperty(_context = "".concat(projectKey, "/")).call(_context, OPERATIONS_EXPORT_RESOURCES_MODAL_SELECTION), {}),
|
|
551
|
+
_useStorage2 = _slicedToArray(_useStorage, 1),
|
|
552
|
+
cachedSelectedGroups = _useStorage2[0];
|
|
528
553
|
return React.useMemo(() => {
|
|
529
|
-
var
|
|
554
|
+
var _context2;
|
|
555
|
+
const groups = updateFieldGroupsWithIsCheckedValue(props.fieldGroups, cachedSelectedGroups[props.resourceType] || {});
|
|
530
556
|
return {
|
|
531
557
|
outputFormat: props.outputFormat,
|
|
532
|
-
fileName: _concatInstanceProperty(
|
|
558
|
+
fileName: _concatInstanceProperty(_context2 = "".concat(resourceTypeToDisplayName({
|
|
533
559
|
resourceType: props.resourceType,
|
|
534
560
|
isUpperCase: true,
|
|
535
561
|
isPlural: true
|
|
536
|
-
}), "_Export_")).call(
|
|
562
|
+
}), "_Export_")).call(_context2, todayFormatted),
|
|
537
563
|
locales: [locale],
|
|
538
564
|
groups: updateFieldGroupWithAdditionalFieldExtensions(groups)
|
|
539
565
|
};
|
|
540
|
-
}, [props,
|
|
566
|
+
}, [props.fieldGroups, props.outputFormat, props.resourceType, cachedSelectedGroups, todayFormatted, locale]);
|
|
541
567
|
};
|
|
542
568
|
|
|
543
569
|
async function createExportOperation(_ref) {
|
|
@@ -816,7 +842,7 @@ const NoSearchResults = () => {
|
|
|
816
842
|
};
|
|
817
843
|
|
|
818
844
|
function ownKeys$1(e, r) { var t = _Object$keys(e); if (_Object$getOwnPropertySymbols) { var o = _Object$getOwnPropertySymbols(e); r && (o = _filterInstanceProperty(o).call(o, function (r) { return _Object$getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
|
|
819
|
-
function _objectSpread$1(e) { for (var r = 1; r < arguments.length; r++) { var
|
|
845
|
+
function _objectSpread$1(e) { for (var r = 1; r < arguments.length; r++) { var _context11, _context12; var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? _forEachInstanceProperty(_context11 = ownKeys$1(Object(t), !0)).call(_context11, function (r) { _defineProperty(e, r, t[r]); }) : _Object$getOwnPropertyDescriptors ? _Object$defineProperties(e, _Object$getOwnPropertyDescriptors(t)) : _forEachInstanceProperty(_context12 = ownKeys$1(Object(t))).call(_context12, function (r) { _Object$defineProperty(e, r, _Object$getOwnPropertyDescriptor(t, r)); }); } return e; }
|
|
820
846
|
function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
|
|
821
847
|
const StyledGridCard = /*#__PURE__*/_styled(Card, process.env.NODE_ENV === "production" ? {
|
|
822
848
|
target: "e3xiyhb2"
|
|
@@ -829,7 +855,7 @@ const StyledGridCard = /*#__PURE__*/_styled(Card, process.env.NODE_ENV === "prod
|
|
|
829
855
|
} : {
|
|
830
856
|
name: "1lf8z6",
|
|
831
857
|
styles: "max-height:50vh;overflow:auto",
|
|
832
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["export-fields-selection-step.tsx"],"names":[],"mappings":"AAwBmC","file":"export-fields-selection-step.tsx","sourcesContent":["import React from 'react'\nimport { useIntl } from 'react-intl'\nimport throttle from 'lodash/throttle'\nimport styled from '@emotion/styled'\nimport { css } from '@emotion/react'\nimport { FormDialog } from '@commercetools-frontend/application-components'\nimport {\n  Card,\n  CheckboxInput,\n  Grid,\n  Link,\n  Spacings,\n  Text,\n  SearchTextInput,\n  designTokens,\n} from '@commercetools-frontend/ui-kit'\nimport get from 'lodash/get'\nimport messages from '../messages'\nimport { useExportResourcesContext } from '../@hooks'\nimport { FIELD_DEFINITIONS_URLS } from '../@constants'\nimport { resourceTypeToDisplayName } from '../@utils'\nimport { NoSearchResults } from './no-search-results'\nimport { FieldGroup, FormikField } from '@types'\n\nconst StyledGridCard = styled(Card)`\n  max-height: 50vh;\n  overflow: auto;\n`\n\nconst StyledGroupCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-right: ${designTokens.borderWidth1} solid\n    ${designTokens.colorNeutral90};\n  background-color: ${designTokens.colorNeutral98};\n  height: 100%;\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\nconst StyledFieldsCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\ninterface FilteredField extends FormikField {\n  isHidden?: boolean\n}\ninterface FilteredGroups extends FieldGroup {\n  isHidden?: boolean\n  fields: FilteredField[]\n}\n\ntype FieldSectionsProps = {\n  fields?: FilteredField[]\n  parentKeys: string[]\n  formik: any\n  isAnySiblingChecked: boolean\n}\n\nconst FieldSections = ({\n  fields,\n  parentKeys,\n  formik,\n  isAnySiblingChecked,\n}: FieldSectionsProps) => {\n  return (\n    <Spacings.Stack scale=\"s\">\n      {fields?.map((field, index) => {\n        const isAtleastOneChildChecked = () => {\n          function checkAtLeastOneFieldChecked(fields): boolean {\n            return fields?.some((field) => {\n              if (field.fields) {\n                return (\n                  field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n                )\n              }\n              return field.isChecked\n            })\n          }\n\n          return checkAtLeastOneFieldChecked(field.fields)\n        }\n        const isAnyOfTheChildFieldsChecked = isAtleastOneChildChecked()\n        if (field.isHidden) return null\n        return (\n          <React.Fragment key={`${field.name}-${index}`}>\n            <CheckboxInput\n              isChecked={\n                field.isReadOnly\n                  ? isAnySiblingChecked\n                  : isAnyOfTheChildFieldsChecked || field.isChecked\n              }\n              isReadOnly={field.isReadOnly || field.isRequired}\n              onChange={(e) => {\n                const path = `${parentKeys.join('.')}.fields[${index}]`\n                // Update all child fields\n                const updateAllChildFieldsChecked = (\n                  path: string,\n                  checked: boolean\n                ) => {\n                  const item = get(formik.values, path)\n                  if (item && item.fields) {\n                    item.fields.forEach((field, i) => {\n                      const fieldPath = `${path}.fields[${i}]`\n                      if (!field.fields?.length) {\n                        formik.setFieldValue(\n                          `${fieldPath}.isChecked`,\n                          checked,\n                          // pass `false` to avoid revalidating the form with intermediate values\n                          false\n                        )\n                      }\n                      updateAllChildFieldsChecked(fieldPath, checked)\n                    })\n                  }\n                }\n                if (!field.fields?.length) {\n                  formik.setFieldValue(\n                    `${path}.isChecked`,\n                    e.target.checked,\n                    // pass `false` to avoid revalidating the form with intermediate values\n                    false\n                  )\n                }\n                updateAllChildFieldsChecked(path, e.target.checked)\n              }}\n            >\n              {field.label} {field.isRequired && '*'}\n            </CheckboxInput>\n            {field.fields && field.fields?.length > 0 && (\n              <Spacings.Inline scale=\"l\">\n                <div />\n                <FieldSections\n                  fields={field.fields}\n                  formik={formik}\n                  parentKeys={[...parentKeys, `fields[${index}]`]}\n                  isAnySiblingChecked={isAnySiblingChecked}\n                />\n              </Spacings.Inline>\n            )}\n          </React.Fragment>\n        )\n      })}\n    </Spacings.Stack>\n  )\n}\n\ntype FieldGroupSectionProps = {\n  index: number\n  group: FilteredGroups\n  formik: any\n}\n\nfunction checkAllFieldsSelected(fields): boolean {\n  return fields.every((field) => {\n    if (field.fields) {\n      return checkAllFieldsSelected(field.fields)\n    }\n    return field.isChecked || field.isRequired || field.isReadOnly\n  })\n}\n\nfunction checkAtLeastOneFieldChecked(fields): boolean {\n  return fields.some((field) => {\n    if (field.fields) {\n      return field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n    }\n    return field.isChecked\n  })\n}\n\nconst FieldGroupSection = ({\n  group,\n  index,\n  formik,\n}: FieldGroupSectionProps) => {\n  const allAllFieldsSelected = React.useMemo(\n    () => checkAllFieldsSelected(group.fields),\n    [group.fields]\n  )\n\n  const isAtleastOneFieldChecked = React.useMemo(\n    () => checkAtLeastOneFieldChecked(group.fields),\n    [group.fields]\n  )\n\n  const parentKey = `groups[${index}]`\n\n  const handleChangeAllFields = (e) => {\n    const item = get(formik.values, parentKey)\n    function updateAllFields(fields, checked) {\n      return fields.map((field) => {\n        if (field.fields) {\n          return {\n            ...field,\n            fields: updateAllFields(field.fields, checked),\n          }\n        }\n        if (field.isReadOnly || field.isRequired) return field\n        return {\n          ...field,\n          isChecked: checked,\n        }\n      })\n    }\n\n    formik.setFieldValue(parentKey, {\n      ...item,\n      fields: updateAllFields(item.fields, e.target.checked),\n    })\n  }\n\n  if (group.isHidden) return null\n  return (\n    <>\n      <Grid.Item>\n        <StyledGroupCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <CheckboxInput\n            isIndeterminate={!allAllFieldsSelected && isAtleastOneFieldChecked}\n            isChecked={allAllFieldsSelected}\n            onChange={handleChangeAllFields}\n          >\n            {group.groupLabel}\n          </CheckboxInput>\n        </StyledGroupCard>\n      </Grid.Item>\n      <Grid.Item>\n        <StyledFieldsCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <Spacings.Stack scale=\"s\">\n            <FieldSections\n              fields={group.fields}\n              formik={formik}\n              parentKeys={[parentKey]}\n              isAnySiblingChecked={isAtleastOneFieldChecked}\n            />\n          </Spacings.Stack>\n        </StyledFieldsCard>\n      </Grid.Item>\n    </>\n  )\n}\n\nexport const ExportFieldsSelectionStep = () => {\n  const intl = useIntl()\n  const [fieldSearchTerm, setFieldSearchTerm] = React.useState('')\n  const { formik, setCurrentStep, resourceType, onClose } =\n    useExportResourcesContext()\n  const resourceTypePlural = resourceTypeToDisplayName({\n    resourceType: resourceType,\n    isPlural: true,\n  })\n\n  React.useEffect(() => {\n    formik.validateForm()\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [formik.values.groups])\n  // adds `isHidden` property to each field based on search term\n  const filteredGroups = React.useMemo(() => {\n    const updateFields = (fields, parentMatches: boolean) => {\n      return fields.map((field) => {\n        const matches = field.label\n          .toLowerCase()\n          .includes(fieldSearchTerm.toLowerCase())\n        let isHidden = !matches && !parentMatches\n        if (field.fields) {\n          const updatedFields = updateFields(field.fields, matches)\n          isHidden = isHidden && updatedFields.every((f) => f.isHidden)\n          field = {\n            ...field,\n            fields: updatedFields,\n          }\n        }\n        return { ...field, isHidden }\n      })\n    }\n    if (!fieldSearchTerm?.trim()?.length) return formik.values.groups\n\n    return formik.values.groups.map((group) => {\n      const updatedFields = updateFields(group.fields, false)\n      const isHidden = updatedFields.every((field) => field.isHidden)\n      return { ...group, fields: updatedFields, isHidden }\n    })\n  }, [formik.values.groups, fieldSearchTerm])\n\n  const onSearchChange = React.useCallback(\n    (event) => {\n      setFieldSearchTerm(event.target.value)\n    },\n    [setFieldSearchTerm]\n  )\n\n  const hasResults = Boolean(filteredGroups.find((group) => !group.isHidden))\n\n  const onSearchChangeThrottled = React.useMemo(\n    () => throttle(onSearchChange, 300),\n    [onSearchChange]\n  )\n\n  const onSearchReset = () => {\n    setFieldSearchTerm('')\n  }\n\n  return (\n    <FormDialog\n      isOpen\n      title={intl.formatMessage(messages.modalTitle, {\n        resourceType: resourceTypePlural,\n      })}\n      labelPrimary={intl.formatMessage(messages.startExport)}\n      onPrimaryButtonClick={formik.submitForm}\n      isPrimaryButtonDisabled={!formik.isValid}\n      labelSecondary={intl.formatMessage(messages.back)}\n      onSecondaryButtonClick={() => setCurrentStep(1)}\n      onClose={onClose}\n      size={16}\n    >\n      <Spacings.Stack scale=\"xl\">\n        <Spacings.Stack scale=\"m\">\n          <Spacings.Inline justifyContent=\"space-between\">\n            <Text.Body intlMessage={messages.selectFieldsToExport} />\n            <Link\n              isExternal\n              tone=\"secondary\"\n              intlMessage={messages.fieldDefinitions}\n              to={FIELD_DEFINITIONS_URLS[resourceType]}\n            />\n          </Spacings.Inline>\n          <SearchTextInput\n            placeholder={intl.formatMessage(messages.searchForFields)}\n            value={fieldSearchTerm}\n            onChange={onSearchChangeThrottled}\n            onReset={onSearchReset}\n            // TODO: `SearchTextInput` props to be improved. The `onSubmit` callback shouldn't be mandatory\n            onSubmit={() => {}}\n          />\n          {hasResults ? (\n            <StyledGridCard insetScale=\"none\">\n              <Grid\n                gridAutoFlow=\"row\"\n                gridTemplateColumns={`${designTokens.constraint5} auto`}\n              >\n                {filteredGroups.map((group, index) => (\n                  <FieldGroupSection\n                    key={group.groupLabel}\n                    group={group}\n                    index={index}\n                    formik={formik}\n                  />\n                ))}\n              </Grid>\n            </StyledGridCard>\n          ) : (\n            <NoSearchResults />\n          )}\n        </Spacings.Stack>\n      </Spacings.Stack>\n    </FormDialog>\n  )\n}\n"]} */",
|
|
858
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["export-fields-selection-step.tsx"],"names":[],"mappings":"AA6BmC","file":"export-fields-selection-step.tsx","sourcesContent":["import React from 'react'\nimport { useIntl } from 'react-intl'\nimport throttle from 'lodash/throttle'\nimport styled from '@emotion/styled'\nimport { css } from '@emotion/react'\nimport { FormDialog } from '@commercetools-frontend/application-components'\nimport {\n  Card,\n  CheckboxInput,\n  Grid,\n  Link,\n  Spacings,\n  Text,\n  SearchTextInput,\n  designTokens,\n} from '@commercetools-frontend/ui-kit'\nimport get from 'lodash/get'\nimport messages from '../messages'\nimport { useExportResourcesContext } from '../@hooks'\nimport {\n  OPERATIONS_EXPORT_RESOURCES_MODAL_SELECTION,\n  FIELD_DEFINITIONS_URLS,\n} from '../@constants'\nimport { resourceTypeToDisplayName } from '../@utils'\nimport { NoSearchResults } from './no-search-results'\nimport { FieldGroup, FormikField } from '@types'\nimport { useStorage } from '@commercetools-frontend/experimental-components'\nimport { useApplicationContext } from '@commercetools-frontend/application-shell-connectors'\n\nconst StyledGridCard = styled(Card)`\n  max-height: 50vh;\n  overflow: auto;\n`\n\nconst StyledGroupCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-right: ${designTokens.borderWidth1} solid\n    ${designTokens.colorNeutral90};\n  background-color: ${designTokens.colorNeutral98};\n  height: 100%;\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\nconst StyledFieldsCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\ninterface FilteredField extends FormikField {\n  isHidden?: boolean\n}\ninterface FilteredGroups extends FieldGroup {\n  isHidden?: boolean\n  fields: FilteredField[]\n}\n\ntype FieldSectionsProps = {\n  fields?: FilteredField[]\n  parentKeys: string[]\n  formik: any\n  isAnySiblingChecked: boolean\n}\n\nconst FieldSections = ({\n  fields,\n  parentKeys,\n  formik,\n  isAnySiblingChecked,\n}: FieldSectionsProps) => {\n  return (\n    <Spacings.Stack scale=\"s\">\n      {fields?.map((field, index) => {\n        const isAtleastOneChildChecked = () => {\n          function checkAtLeastOneFieldChecked(fields): boolean {\n            return fields?.some((field) => {\n              if (field.fields) {\n                return (\n                  field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n                )\n              }\n              return field.isChecked\n            })\n          }\n\n          return checkAtLeastOneFieldChecked(field.fields)\n        }\n        const isAnyOfTheChildFieldsChecked = isAtleastOneChildChecked()\n        if (field.isHidden) return null\n        return (\n          <React.Fragment key={`${field.name}-${index}`}>\n            <CheckboxInput\n              isChecked={\n                field.isReadOnly\n                  ? isAnySiblingChecked\n                  : isAnyOfTheChildFieldsChecked || field.isChecked\n              }\n              isReadOnly={field.isReadOnly || field.isRequired}\n              onChange={(e) => {\n                const path = `${parentKeys.join('.')}.fields[${index}]`\n                // Update all child fields\n                const updateAllChildFieldsChecked = (\n                  path: string,\n                  checked: boolean\n                ) => {\n                  const item = get(formik.values, path)\n                  if (item && item.fields) {\n                    item.fields.forEach((field, i) => {\n                      const fieldPath = `${path}.fields[${i}]`\n                      if (!field.fields?.length) {\n                        formik.setFieldValue(\n                          `${fieldPath}.isChecked`,\n                          checked,\n                          // pass `false` to avoid revalidating the form with intermediate values\n                          false\n                        )\n                      }\n                      updateAllChildFieldsChecked(fieldPath, checked)\n                    })\n                  }\n                }\n                if (!field.fields?.length) {\n                  formik.setFieldValue(\n                    `${path}.isChecked`,\n                    e.target.checked,\n                    // pass `false` to avoid revalidating the form with intermediate values\n                    false\n                  )\n                }\n                updateAllChildFieldsChecked(path, e.target.checked)\n              }}\n            >\n              {field.label} {field.isRequired && '*'}\n            </CheckboxInput>\n            {field.fields && field.fields?.length > 0 && (\n              <Spacings.Inline scale=\"l\">\n                <div />\n                <FieldSections\n                  fields={field.fields}\n                  formik={formik}\n                  parentKeys={[...parentKeys, `fields[${index}]`]}\n                  isAnySiblingChecked={isAnySiblingChecked}\n                />\n              </Spacings.Inline>\n            )}\n          </React.Fragment>\n        )\n      })}\n    </Spacings.Stack>\n  )\n}\n\ntype FieldGroupSectionProps = {\n  index: number\n  group: FilteredGroups\n  formik: any\n}\n\nfunction checkAllFieldsSelected(fields): boolean {\n  return fields.every((field) => {\n    if (field.fields) {\n      return checkAllFieldsSelected(field.fields)\n    }\n    return field.isChecked || field.isRequired || field.isReadOnly\n  })\n}\n\nfunction checkAtLeastOneFieldChecked(fields): boolean {\n  return fields.some((field) => {\n    if (field.fields) {\n      return field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n    }\n    return field.isChecked\n  })\n}\n\nconst FieldGroupSection = ({\n  group,\n  index,\n  formik,\n}: FieldGroupSectionProps) => {\n  const allAllFieldsSelected = React.useMemo(\n    () => checkAllFieldsSelected(group.fields),\n    [group.fields]\n  )\n\n  const isAtleastOneFieldChecked = React.useMemo(\n    () => checkAtLeastOneFieldChecked(group.fields),\n    [group.fields]\n  )\n\n  const parentKey = `groups[${index}]`\n\n  const handleChangeAllFields = (e) => {\n    const item = get(formik.values, parentKey)\n    function updateAllFields(fields, checked) {\n      return fields.map((field) => {\n        if (field.fields) {\n          return {\n            ...field,\n            fields: updateAllFields(field.fields, checked),\n          }\n        }\n        if (field.isReadOnly || field.isRequired) return field\n        return {\n          ...field,\n          isChecked: checked,\n        }\n      })\n    }\n\n    formik.setFieldValue(parentKey, {\n      ...item,\n      fields: updateAllFields(item.fields, e.target.checked),\n    })\n  }\n\n  if (group.isHidden) return null\n  return (\n    <>\n      <Grid.Item>\n        <StyledGroupCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <CheckboxInput\n            isIndeterminate={!allAllFieldsSelected && isAtleastOneFieldChecked}\n            isChecked={allAllFieldsSelected}\n            onChange={handleChangeAllFields}\n          >\n            {group.groupLabel}\n          </CheckboxInput>\n        </StyledGroupCard>\n      </Grid.Item>\n      <Grid.Item>\n        <StyledFieldsCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <Spacings.Stack scale=\"s\">\n            <FieldSections\n              fields={group.fields}\n              formik={formik}\n              parentKeys={[parentKey]}\n              isAnySiblingChecked={isAtleastOneFieldChecked}\n            />\n          </Spacings.Stack>\n        </StyledFieldsCard>\n      </Grid.Item>\n    </>\n  )\n}\n\nexport const ExportFieldsSelectionStep = () => {\n  const intl = useIntl()\n  const [fieldSearchTerm, setFieldSearchTerm] = React.useState('')\n  const { projectKey } = useApplicationContext((context) => ({\n    projectKey: context.project?.key,\n  }))\n  const [_, setCachedSelectedGroups] = useStorage(\n    `${projectKey}/${OPERATIONS_EXPORT_RESOURCES_MODAL_SELECTION}`,\n    {}\n  )\n  const { formik, setCurrentStep, resourceType, onClose } =\n    useExportResourcesContext()\n  const resourceTypePlural = resourceTypeToDisplayName({\n    resourceType: resourceType,\n    isPlural: true,\n  })\n\n  React.useEffect(() => {\n    formik.validateForm()\n    const selectedFields = {}\n    function getSelectedFields(field, groupName) {\n      const fieldName = field.name ? `${groupName}.${field.name}` : groupName\n      if (field.fields) {\n        return field.fields.forEach((f) => getSelectedFields(f, fieldName))\n      }\n\n      selectedFields[fieldName] = field.isChecked\n    }\n    formik.values.groups.forEach((group) =>\n      getSelectedFields(group, group.groupLabel)\n    )\n    setCachedSelectedGroups({\n      [resourceType]: selectedFields,\n    })\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [formik.values.groups])\n  // adds `isHidden` property to each field based on search term\n  const filteredGroups = React.useMemo(() => {\n    const updateFields = (fields, parentMatches: boolean) => {\n      return fields.map((field) => {\n        const matches = field.label\n          .toLowerCase()\n          .includes(fieldSearchTerm.toLowerCase())\n        let isHidden = !matches && !parentMatches\n        if (field.fields) {\n          const updatedFields = updateFields(field.fields, matches)\n          isHidden = isHidden && updatedFields.every((f) => f.isHidden)\n          field = {\n            ...field,\n            fields: updatedFields,\n          }\n        }\n        return { ...field, isHidden }\n      })\n    }\n    if (!fieldSearchTerm?.trim()?.length) return formik.values.groups\n\n    return formik.values.groups.map((group) => {\n      const updatedFields = updateFields(group.fields, false)\n      const isHidden = updatedFields.every((field) => field.isHidden)\n      return { ...group, fields: updatedFields, isHidden }\n    })\n  }, [formik.values.groups, fieldSearchTerm])\n\n  const onSearchChange = React.useCallback(\n    (event) => {\n      setFieldSearchTerm(event.target.value)\n    },\n    [setFieldSearchTerm]\n  )\n\n  const hasResults = Boolean(filteredGroups.find((group) => !group.isHidden))\n\n  const onSearchChangeThrottled = React.useMemo(\n    () => throttle(onSearchChange, 300),\n    [onSearchChange]\n  )\n\n  const onSearchReset = () => {\n    setFieldSearchTerm('')\n  }\n\n  return (\n    <FormDialog\n      isOpen\n      title={intl.formatMessage(messages.modalTitle, {\n        resourceType: resourceTypePlural,\n      })}\n      labelPrimary={intl.formatMessage(messages.startExport)}\n      onPrimaryButtonClick={formik.submitForm}\n      isPrimaryButtonDisabled={!formik.isValid}\n      labelSecondary={intl.formatMessage(messages.back)}\n      onSecondaryButtonClick={() => setCurrentStep(1)}\n      onClose={onClose}\n      size={16}\n    >\n      <Spacings.Stack scale=\"xl\">\n        <Spacings.Stack scale=\"m\">\n          <Spacings.Inline justifyContent=\"space-between\">\n            <Text.Body intlMessage={messages.selectFieldsToExport} />\n            <Link\n              isExternal\n              tone=\"secondary\"\n              intlMessage={messages.fieldDefinitions}\n              to={FIELD_DEFINITIONS_URLS[resourceType]}\n            />\n          </Spacings.Inline>\n          <SearchTextInput\n            placeholder={intl.formatMessage(messages.searchForFields)}\n            value={fieldSearchTerm}\n            onChange={onSearchChangeThrottled}\n            onReset={onSearchReset}\n            // TODO: `SearchTextInput` props to be improved. The `onSubmit` callback shouldn't be mandatory\n            onSubmit={() => {}}\n          />\n          {hasResults ? (\n            <StyledGridCard insetScale=\"none\">\n              <Grid\n                gridAutoFlow=\"row\"\n                gridTemplateColumns={`${designTokens.constraint5} auto`}\n              >\n                {filteredGroups.map((group, index) => (\n                  <FieldGroupSection\n                    key={group.groupLabel}\n                    group={group}\n                    index={index}\n                    formik={formik}\n                  />\n                ))}\n              </Grid>\n            </StyledGridCard>\n          ) : (\n            <NoSearchResults />\n          )}\n        </Spacings.Stack>\n      </Spacings.Stack>\n    </FormDialog>\n  )\n}\n"]} */",
|
|
833
859
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
834
860
|
});
|
|
835
861
|
const StyledGroupCard = /*#__PURE__*/_styled(Card, process.env.NODE_ENV === "production" ? {
|
|
@@ -837,13 +863,13 @@ const StyledGroupCard = /*#__PURE__*/_styled(Card, process.env.NODE_ENV === "pro
|
|
|
837
863
|
} : {
|
|
838
864
|
target: "e3xiyhb1",
|
|
839
865
|
label: "StyledGroupCard"
|
|
840
|
-
})("border-right:", designTokens.borderWidth1, " solid ", designTokens.colorNeutral90, ";background-color:", designTokens.colorNeutral98, ";height:100%;border-radius:0;", props => props.showTopBorder && /*#__PURE__*/css("border-top:", designTokens.borderWidth1, " solid ", designTokens.colorNeutral90, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:StyledGroupCard;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["export-fields-selection-step.tsx"],"names":[],"mappings":"AAqCO","file":"export-fields-selection-step.tsx","sourcesContent":["import React from 'react'\nimport { useIntl } from 'react-intl'\nimport throttle from 'lodash/throttle'\nimport styled from '@emotion/styled'\nimport { css } from '@emotion/react'\nimport { FormDialog } from '@commercetools-frontend/application-components'\nimport {\n  Card,\n  CheckboxInput,\n  Grid,\n  Link,\n  Spacings,\n  Text,\n  SearchTextInput,\n  designTokens,\n} from '@commercetools-frontend/ui-kit'\nimport get from 'lodash/get'\nimport messages from '../messages'\nimport { useExportResourcesContext } from '../@hooks'\nimport { FIELD_DEFINITIONS_URLS } from '../@constants'\nimport { resourceTypeToDisplayName } from '../@utils'\nimport { NoSearchResults } from './no-search-results'\nimport { FieldGroup, FormikField } from '@types'\n\nconst StyledGridCard = styled(Card)`\n  max-height: 50vh;\n  overflow: auto;\n`\n\nconst StyledGroupCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-right: ${designTokens.borderWidth1} solid\n    ${designTokens.colorNeutral90};\n  background-color: ${designTokens.colorNeutral98};\n  height: 100%;\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\nconst StyledFieldsCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\ninterface FilteredField extends FormikField {\n  isHidden?: boolean\n}\ninterface FilteredGroups extends FieldGroup {\n  isHidden?: boolean\n  fields: FilteredField[]\n}\n\ntype FieldSectionsProps = {\n  fields?: FilteredField[]\n  parentKeys: string[]\n  formik: any\n  isAnySiblingChecked: boolean\n}\n\nconst FieldSections = ({\n  fields,\n  parentKeys,\n  formik,\n  isAnySiblingChecked,\n}: FieldSectionsProps) => {\n  return (\n    <Spacings.Stack scale=\"s\">\n      {fields?.map((field, index) => {\n        const isAtleastOneChildChecked = () => {\n          function checkAtLeastOneFieldChecked(fields): boolean {\n            return fields?.some((field) => {\n              if (field.fields) {\n                return (\n                  field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n                )\n              }\n              return field.isChecked\n            })\n          }\n\n          return checkAtLeastOneFieldChecked(field.fields)\n        }\n        const isAnyOfTheChildFieldsChecked = isAtleastOneChildChecked()\n        if (field.isHidden) return null\n        return (\n          <React.Fragment key={`${field.name}-${index}`}>\n            <CheckboxInput\n              isChecked={\n                field.isReadOnly\n                  ? isAnySiblingChecked\n                  : isAnyOfTheChildFieldsChecked || field.isChecked\n              }\n              isReadOnly={field.isReadOnly || field.isRequired}\n              onChange={(e) => {\n                const path = `${parentKeys.join('.')}.fields[${index}]`\n                // Update all child fields\n                const updateAllChildFieldsChecked = (\n                  path: string,\n                  checked: boolean\n                ) => {\n                  const item = get(formik.values, path)\n                  if (item && item.fields) {\n                    item.fields.forEach((field, i) => {\n                      const fieldPath = `${path}.fields[${i}]`\n                      if (!field.fields?.length) {\n                        formik.setFieldValue(\n                          `${fieldPath}.isChecked`,\n                          checked,\n                          // pass `false` to avoid revalidating the form with intermediate values\n                          false\n                        )\n                      }\n                      updateAllChildFieldsChecked(fieldPath, checked)\n                    })\n                  }\n                }\n                if (!field.fields?.length) {\n                  formik.setFieldValue(\n                    `${path}.isChecked`,\n                    e.target.checked,\n                    // pass `false` to avoid revalidating the form with intermediate values\n                    false\n                  )\n                }\n                updateAllChildFieldsChecked(path, e.target.checked)\n              }}\n            >\n              {field.label} {field.isRequired && '*'}\n            </CheckboxInput>\n            {field.fields && field.fields?.length > 0 && (\n              <Spacings.Inline scale=\"l\">\n                <div />\n                <FieldSections\n                  fields={field.fields}\n                  formik={formik}\n                  parentKeys={[...parentKeys, `fields[${index}]`]}\n                  isAnySiblingChecked={isAnySiblingChecked}\n                />\n              </Spacings.Inline>\n            )}\n          </React.Fragment>\n        )\n      })}\n    </Spacings.Stack>\n  )\n}\n\ntype FieldGroupSectionProps = {\n  index: number\n  group: FilteredGroups\n  formik: any\n}\n\nfunction checkAllFieldsSelected(fields): boolean {\n  return fields.every((field) => {\n    if (field.fields) {\n      return checkAllFieldsSelected(field.fields)\n    }\n    return field.isChecked || field.isRequired || field.isReadOnly\n  })\n}\n\nfunction checkAtLeastOneFieldChecked(fields): boolean {\n  return fields.some((field) => {\n    if (field.fields) {\n      return field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n    }\n    return field.isChecked\n  })\n}\n\nconst FieldGroupSection = ({\n  group,\n  index,\n  formik,\n}: FieldGroupSectionProps) => {\n  const allAllFieldsSelected = React.useMemo(\n    () => checkAllFieldsSelected(group.fields),\n    [group.fields]\n  )\n\n  const isAtleastOneFieldChecked = React.useMemo(\n    () => checkAtLeastOneFieldChecked(group.fields),\n    [group.fields]\n  )\n\n  const parentKey = `groups[${index}]`\n\n  const handleChangeAllFields = (e) => {\n    const item = get(formik.values, parentKey)\n    function updateAllFields(fields, checked) {\n      return fields.map((field) => {\n        if (field.fields) {\n          return {\n            ...field,\n            fields: updateAllFields(field.fields, checked),\n          }\n        }\n        if (field.isReadOnly || field.isRequired) return field\n        return {\n          ...field,\n          isChecked: checked,\n        }\n      })\n    }\n\n    formik.setFieldValue(parentKey, {\n      ...item,\n      fields: updateAllFields(item.fields, e.target.checked),\n    })\n  }\n\n  if (group.isHidden) return null\n  return (\n    <>\n      <Grid.Item>\n        <StyledGroupCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <CheckboxInput\n            isIndeterminate={!allAllFieldsSelected && isAtleastOneFieldChecked}\n            isChecked={allAllFieldsSelected}\n            onChange={handleChangeAllFields}\n          >\n            {group.groupLabel}\n          </CheckboxInput>\n        </StyledGroupCard>\n      </Grid.Item>\n      <Grid.Item>\n        <StyledFieldsCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <Spacings.Stack scale=\"s\">\n            <FieldSections\n              fields={group.fields}\n              formik={formik}\n              parentKeys={[parentKey]}\n              isAnySiblingChecked={isAtleastOneFieldChecked}\n            />\n          </Spacings.Stack>\n        </StyledFieldsCard>\n      </Grid.Item>\n    </>\n  )\n}\n\nexport const ExportFieldsSelectionStep = () => {\n  const intl = useIntl()\n  const [fieldSearchTerm, setFieldSearchTerm] = React.useState('')\n  const { formik, setCurrentStep, resourceType, onClose } =\n    useExportResourcesContext()\n  const resourceTypePlural = resourceTypeToDisplayName({\n    resourceType: resourceType,\n    isPlural: true,\n  })\n\n  React.useEffect(() => {\n    formik.validateForm()\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [formik.values.groups])\n  // adds `isHidden` property to each field based on search term\n  const filteredGroups = React.useMemo(() => {\n    const updateFields = (fields, parentMatches: boolean) => {\n      return fields.map((field) => {\n        const matches = field.label\n          .toLowerCase()\n          .includes(fieldSearchTerm.toLowerCase())\n        let isHidden = !matches && !parentMatches\n        if (field.fields) {\n          const updatedFields = updateFields(field.fields, matches)\n          isHidden = isHidden && updatedFields.every((f) => f.isHidden)\n          field = {\n            ...field,\n            fields: updatedFields,\n          }\n        }\n        return { ...field, isHidden }\n      })\n    }\n    if (!fieldSearchTerm?.trim()?.length) return formik.values.groups\n\n    return formik.values.groups.map((group) => {\n      const updatedFields = updateFields(group.fields, false)\n      const isHidden = updatedFields.every((field) => field.isHidden)\n      return { ...group, fields: updatedFields, isHidden }\n    })\n  }, [formik.values.groups, fieldSearchTerm])\n\n  const onSearchChange = React.useCallback(\n    (event) => {\n      setFieldSearchTerm(event.target.value)\n    },\n    [setFieldSearchTerm]\n  )\n\n  const hasResults = Boolean(filteredGroups.find((group) => !group.isHidden))\n\n  const onSearchChangeThrottled = React.useMemo(\n    () => throttle(onSearchChange, 300),\n    [onSearchChange]\n  )\n\n  const onSearchReset = () => {\n    setFieldSearchTerm('')\n  }\n\n  return (\n    <FormDialog\n      isOpen\n      title={intl.formatMessage(messages.modalTitle, {\n        resourceType: resourceTypePlural,\n      })}\n      labelPrimary={intl.formatMessage(messages.startExport)}\n      onPrimaryButtonClick={formik.submitForm}\n      isPrimaryButtonDisabled={!formik.isValid}\n      labelSecondary={intl.formatMessage(messages.back)}\n      onSecondaryButtonClick={() => setCurrentStep(1)}\n      onClose={onClose}\n      size={16}\n    >\n      <Spacings.Stack scale=\"xl\">\n        <Spacings.Stack scale=\"m\">\n          <Spacings.Inline justifyContent=\"space-between\">\n            <Text.Body intlMessage={messages.selectFieldsToExport} />\n            <Link\n              isExternal\n              tone=\"secondary\"\n              intlMessage={messages.fieldDefinitions}\n              to={FIELD_DEFINITIONS_URLS[resourceType]}\n            />\n          </Spacings.Inline>\n          <SearchTextInput\n            placeholder={intl.formatMessage(messages.searchForFields)}\n            value={fieldSearchTerm}\n            onChange={onSearchChangeThrottled}\n            onReset={onSearchReset}\n            // TODO: `SearchTextInput` props to be improved. The `onSubmit` callback shouldn't be mandatory\n            onSubmit={() => {}}\n          />\n          {hasResults ? (\n            <StyledGridCard insetScale=\"none\">\n              <Grid\n                gridAutoFlow=\"row\"\n                gridTemplateColumns={`${designTokens.constraint5} auto`}\n              >\n                {filteredGroups.map((group, index) => (\n                  <FieldGroupSection\n                    key={group.groupLabel}\n                    group={group}\n                    index={index}\n                    formik={formik}\n                  />\n                ))}\n              </Grid>\n            </StyledGridCard>\n          ) : (\n            <NoSearchResults />\n          )}\n        </Spacings.Stack>\n      </Spacings.Stack>\n    </FormDialog>\n  )\n}\n"]} */"), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["export-fields-selection-step.tsx"],"names":[],"mappings":"AA6BiE","file":"export-fields-selection-step.tsx","sourcesContent":["import React from 'react'\nimport { useIntl } from 'react-intl'\nimport throttle from 'lodash/throttle'\nimport styled from '@emotion/styled'\nimport { css } from '@emotion/react'\nimport { FormDialog } from '@commercetools-frontend/application-components'\nimport {\n  Card,\n  CheckboxInput,\n  Grid,\n  Link,\n  Spacings,\n  Text,\n  SearchTextInput,\n  designTokens,\n} from '@commercetools-frontend/ui-kit'\nimport get from 'lodash/get'\nimport messages from '../messages'\nimport { useExportResourcesContext } from '../@hooks'\nimport { FIELD_DEFINITIONS_URLS } from '../@constants'\nimport { resourceTypeToDisplayName } from '../@utils'\nimport { NoSearchResults } from './no-search-results'\nimport { FieldGroup, FormikField } from '@types'\n\nconst StyledGridCard = styled(Card)`\n  max-height: 50vh;\n  overflow: auto;\n`\n\nconst StyledGroupCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-right: ${designTokens.borderWidth1} solid\n    ${designTokens.colorNeutral90};\n  background-color: ${designTokens.colorNeutral98};\n  height: 100%;\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\nconst StyledFieldsCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\ninterface FilteredField extends FormikField {\n  isHidden?: boolean\n}\ninterface FilteredGroups extends FieldGroup {\n  isHidden?: boolean\n  fields: FilteredField[]\n}\n\ntype FieldSectionsProps = {\n  fields?: FilteredField[]\n  parentKeys: string[]\n  formik: any\n  isAnySiblingChecked: boolean\n}\n\nconst FieldSections = ({\n  fields,\n  parentKeys,\n  formik,\n  isAnySiblingChecked,\n}: FieldSectionsProps) => {\n  return (\n    <Spacings.Stack scale=\"s\">\n      {fields?.map((field, index) => {\n        const isAtleastOneChildChecked = () => {\n          function checkAtLeastOneFieldChecked(fields): boolean {\n            return fields?.some((field) => {\n              if (field.fields) {\n                return (\n                  field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n                )\n              }\n              return field.isChecked\n            })\n          }\n\n          return checkAtLeastOneFieldChecked(field.fields)\n        }\n        const isAnyOfTheChildFieldsChecked = isAtleastOneChildChecked()\n        if (field.isHidden) return null\n        return (\n          <React.Fragment key={`${field.name}-${index}`}>\n            <CheckboxInput\n              isChecked={\n                field.isReadOnly\n                  ? isAnySiblingChecked\n                  : isAnyOfTheChildFieldsChecked || field.isChecked\n              }\n              isReadOnly={field.isReadOnly || field.isRequired}\n              onChange={(e) => {\n                const path = `${parentKeys.join('.')}.fields[${index}]`\n                // Update all child fields\n                const updateAllChildFieldsChecked = (\n                  path: string,\n                  checked: boolean\n                ) => {\n                  const item = get(formik.values, path)\n                  if (item && item.fields) {\n                    item.fields.forEach((field, i) => {\n                      const fieldPath = `${path}.fields[${i}]`\n                      if (!field.fields?.length) {\n                        formik.setFieldValue(\n                          `${fieldPath}.isChecked`,\n                          checked,\n                          // pass `false` to avoid revalidating the form with intermediate values\n                          false\n                        )\n                      }\n                      updateAllChildFieldsChecked(fieldPath, checked)\n                    })\n                  }\n                }\n                if (!field.fields?.length) {\n                  formik.setFieldValue(\n                    `${path}.isChecked`,\n                    e.target.checked,\n                    // pass `false` to avoid revalidating the form with intermediate values\n                    false\n                  )\n                }\n                updateAllChildFieldsChecked(path, e.target.checked)\n              }}\n            >\n              {field.label} {field.isRequired && '*'}\n            </CheckboxInput>\n            {field.fields && field.fields?.length > 0 && (\n              <Spacings.Inline scale=\"l\">\n                <div />\n                <FieldSections\n                  fields={field.fields}\n                  formik={formik}\n                  parentKeys={[...parentKeys, `fields[${index}]`]}\n                  isAnySiblingChecked={isAnySiblingChecked}\n                />\n              </Spacings.Inline>\n            )}\n          </React.Fragment>\n        )\n      })}\n    </Spacings.Stack>\n  )\n}\n\ntype FieldGroupSectionProps = {\n  index: number\n  group: FilteredGroups\n  formik: any\n}\n\nfunction checkAllFieldsSelected(fields): boolean {\n  return fields.every((field) => {\n    if (field.fields) {\n      return checkAllFieldsSelected(field.fields)\n    }\n    return field.isChecked || field.isRequired || field.isReadOnly\n  })\n}\n\nfunction checkAtLeastOneFieldChecked(fields): boolean {\n  return fields.some((field) => {\n    if (field.fields) {\n      return field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n    }\n    return field.isChecked\n  })\n}\n\nconst FieldGroupSection = ({\n  group,\n  index,\n  formik,\n}: FieldGroupSectionProps) => {\n  const allAllFieldsSelected = React.useMemo(\n    () => checkAllFieldsSelected(group.fields),\n    [group.fields]\n  )\n\n  const isAtleastOneFieldChecked = React.useMemo(\n    () => checkAtLeastOneFieldChecked(group.fields),\n    [group.fields]\n  )\n\n  const parentKey = `groups[${index}]`\n\n  const handleChangeAllFields = (e) => {\n    const item = get(formik.values, parentKey)\n    function updateAllFields(fields, checked) {\n      return fields.map((field) => {\n        if (field.fields) {\n          return {\n            ...field,\n            fields: updateAllFields(field.fields, checked),\n          }\n        }\n        if (field.isReadOnly || field.isRequired) return field\n        return {\n          ...field,\n          isChecked: checked,\n        }\n      })\n    }\n\n    formik.setFieldValue(parentKey, {\n      ...item,\n      fields: updateAllFields(item.fields, e.target.checked),\n    })\n  }\n\n  if (group.isHidden) return null\n  return (\n    <>\n      <Grid.Item>\n        <StyledGroupCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <CheckboxInput\n            isIndeterminate={!allAllFieldsSelected && isAtleastOneFieldChecked}\n            isChecked={allAllFieldsSelected}\n            onChange={handleChangeAllFields}\n          >\n            {group.groupLabel}\n          </CheckboxInput>\n        </StyledGroupCard>\n      </Grid.Item>\n      <Grid.Item>\n        <StyledFieldsCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <Spacings.Stack scale=\"s\">\n            <FieldSections\n              fields={group.fields}\n              formik={formik}\n              parentKeys={[parentKey]}\n              isAnySiblingChecked={isAtleastOneFieldChecked}\n            />\n          </Spacings.Stack>\n        </StyledFieldsCard>\n      </Grid.Item>\n    </>\n  )\n}\n\nexport const ExportFieldsSelectionStep = () => {\n  const intl = useIntl()\n  const [fieldSearchTerm, setFieldSearchTerm] = React.useState('')\n  const { formik, setCurrentStep, resourceType, onClose } =\n    useExportResourcesContext()\n  const resourceTypePlural = resourceTypeToDisplayName({\n    resourceType: resourceType,\n    isPlural: true,\n  })\n\n  React.useEffect(() => {\n    formik.validateForm()\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [formik.values.groups])\n  // adds `isHidden` property to each field based on search term\n  const filteredGroups = React.useMemo(() => {\n    const updateFields = (fields, parentMatches: boolean) => {\n      return fields.map((field) => {\n        const matches = field.label\n          .toLowerCase()\n          .includes(fieldSearchTerm.toLowerCase())\n        let isHidden = !matches && !parentMatches\n        if (field.fields) {\n          const updatedFields = updateFields(field.fields, matches)\n          isHidden = isHidden && updatedFields.every((f) => f.isHidden)\n          field = {\n            ...field,\n            fields: updatedFields,\n          }\n        }\n        return { ...field, isHidden }\n      })\n    }\n    if (!fieldSearchTerm?.trim()?.length) return formik.values.groups\n\n    return formik.values.groups.map((group) => {\n      const updatedFields = updateFields(group.fields, false)\n      const isHidden = updatedFields.every((field) => field.isHidden)\n      return { ...group, fields: updatedFields, isHidden }\n    })\n  }, [formik.values.groups, fieldSearchTerm])\n\n  const onSearchChange = React.useCallback(\n    (event) => {\n      setFieldSearchTerm(event.target.value)\n    },\n    [setFieldSearchTerm]\n  )\n\n  const hasResults = Boolean(filteredGroups.find((group) => !group.isHidden))\n\n  const onSearchChangeThrottled = React.useMemo(\n    () => throttle(onSearchChange, 300),\n    [onSearchChange]\n  )\n\n  const onSearchReset = () => {\n    setFieldSearchTerm('')\n  }\n\n  return (\n    <FormDialog\n      isOpen\n      title={intl.formatMessage(messages.modalTitle, {\n        resourceType: resourceTypePlural,\n      })}\n      labelPrimary={intl.formatMessage(messages.startExport)}\n      onPrimaryButtonClick={formik.submitForm}\n      isPrimaryButtonDisabled={!formik.isValid}\n      labelSecondary={intl.formatMessage(messages.back)}\n      onSecondaryButtonClick={() => setCurrentStep(1)}\n      onClose={onClose}\n      size={16}\n    >\n      <Spacings.Stack scale=\"xl\">\n        <Spacings.Stack scale=\"m\">\n          <Spacings.Inline justifyContent=\"space-between\">\n            <Text.Body intlMessage={messages.selectFieldsToExport} />\n            <Link\n              isExternal\n              tone=\"secondary\"\n              intlMessage={messages.fieldDefinitions}\n              to={FIELD_DEFINITIONS_URLS[resourceType]}\n            />\n          </Spacings.Inline>\n          <SearchTextInput\n            placeholder={intl.formatMessage(messages.searchForFields)}\n            value={fieldSearchTerm}\n            onChange={onSearchChangeThrottled}\n            onReset={onSearchReset}\n            // TODO: `SearchTextInput` props to be improved. The `onSubmit` callback shouldn't be mandatory\n            onSubmit={() => {}}\n          />\n          {hasResults ? (\n            <StyledGridCard insetScale=\"none\">\n              <Grid\n                gridAutoFlow=\"row\"\n                gridTemplateColumns={`${designTokens.constraint5} auto`}\n              >\n                {filteredGroups.map((group, index) => (\n                  <FieldGroupSection\n                    key={group.groupLabel}\n                    group={group}\n                    index={index}\n                    formik={formik}\n                  />\n                ))}\n              </Grid>\n            </StyledGridCard>\n          ) : (\n            <NoSearchResults />\n          )}\n        </Spacings.Stack>\n      </Spacings.Stack>\n    </FormDialog>\n  )\n}\n"]} */"));
|
|
866
|
+
})("border-right:", designTokens.borderWidth1, " solid ", designTokens.colorNeutral90, ";background-color:", designTokens.colorNeutral98, ";height:100%;border-radius:0;", props => props.showTopBorder && /*#__PURE__*/css("border-top:", designTokens.borderWidth1, " solid ", designTokens.colorNeutral90, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:StyledGroupCard;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["export-fields-selection-step.tsx"],"names":[],"mappings":"AA0CO","file":"export-fields-selection-step.tsx","sourcesContent":["import React from 'react'\nimport { useIntl } from 'react-intl'\nimport throttle from 'lodash/throttle'\nimport styled from '@emotion/styled'\nimport { css } from '@emotion/react'\nimport { FormDialog } from '@commercetools-frontend/application-components'\nimport {\n  Card,\n  CheckboxInput,\n  Grid,\n  Link,\n  Spacings,\n  Text,\n  SearchTextInput,\n  designTokens,\n} from '@commercetools-frontend/ui-kit'\nimport get from 'lodash/get'\nimport messages from '../messages'\nimport { useExportResourcesContext } from '../@hooks'\nimport {\n  OPERATIONS_EXPORT_RESOURCES_MODAL_SELECTION,\n  FIELD_DEFINITIONS_URLS,\n} from '../@constants'\nimport { resourceTypeToDisplayName } from '../@utils'\nimport { NoSearchResults } from './no-search-results'\nimport { FieldGroup, FormikField } from '@types'\nimport { useStorage } from '@commercetools-frontend/experimental-components'\nimport { useApplicationContext } from '@commercetools-frontend/application-shell-connectors'\n\nconst StyledGridCard = styled(Card)`\n  max-height: 50vh;\n  overflow: auto;\n`\n\nconst StyledGroupCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-right: ${designTokens.borderWidth1} solid\n    ${designTokens.colorNeutral90};\n  background-color: ${designTokens.colorNeutral98};\n  height: 100%;\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\nconst StyledFieldsCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\ninterface FilteredField extends FormikField {\n  isHidden?: boolean\n}\ninterface FilteredGroups extends FieldGroup {\n  isHidden?: boolean\n  fields: FilteredField[]\n}\n\ntype FieldSectionsProps = {\n  fields?: FilteredField[]\n  parentKeys: string[]\n  formik: any\n  isAnySiblingChecked: boolean\n}\n\nconst FieldSections = ({\n  fields,\n  parentKeys,\n  formik,\n  isAnySiblingChecked,\n}: FieldSectionsProps) => {\n  return (\n    <Spacings.Stack scale=\"s\">\n      {fields?.map((field, index) => {\n        const isAtleastOneChildChecked = () => {\n          function checkAtLeastOneFieldChecked(fields): boolean {\n            return fields?.some((field) => {\n              if (field.fields) {\n                return (\n                  field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n                )\n              }\n              return field.isChecked\n            })\n          }\n\n          return checkAtLeastOneFieldChecked(field.fields)\n        }\n        const isAnyOfTheChildFieldsChecked = isAtleastOneChildChecked()\n        if (field.isHidden) return null\n        return (\n          <React.Fragment key={`${field.name}-${index}`}>\n            <CheckboxInput\n              isChecked={\n                field.isReadOnly\n                  ? isAnySiblingChecked\n                  : isAnyOfTheChildFieldsChecked || field.isChecked\n              }\n              isReadOnly={field.isReadOnly || field.isRequired}\n              onChange={(e) => {\n                const path = `${parentKeys.join('.')}.fields[${index}]`\n                // Update all child fields\n                const updateAllChildFieldsChecked = (\n                  path: string,\n                  checked: boolean\n                ) => {\n                  const item = get(formik.values, path)\n                  if (item && item.fields) {\n                    item.fields.forEach((field, i) => {\n                      const fieldPath = `${path}.fields[${i}]`\n                      if (!field.fields?.length) {\n                        formik.setFieldValue(\n                          `${fieldPath}.isChecked`,\n                          checked,\n                          // pass `false` to avoid revalidating the form with intermediate values\n                          false\n                        )\n                      }\n                      updateAllChildFieldsChecked(fieldPath, checked)\n                    })\n                  }\n                }\n                if (!field.fields?.length) {\n                  formik.setFieldValue(\n                    `${path}.isChecked`,\n                    e.target.checked,\n                    // pass `false` to avoid revalidating the form with intermediate values\n                    false\n                  )\n                }\n                updateAllChildFieldsChecked(path, e.target.checked)\n              }}\n            >\n              {field.label} {field.isRequired && '*'}\n            </CheckboxInput>\n            {field.fields && field.fields?.length > 0 && (\n              <Spacings.Inline scale=\"l\">\n                <div />\n                <FieldSections\n                  fields={field.fields}\n                  formik={formik}\n                  parentKeys={[...parentKeys, `fields[${index}]`]}\n                  isAnySiblingChecked={isAnySiblingChecked}\n                />\n              </Spacings.Inline>\n            )}\n          </React.Fragment>\n        )\n      })}\n    </Spacings.Stack>\n  )\n}\n\ntype FieldGroupSectionProps = {\n  index: number\n  group: FilteredGroups\n  formik: any\n}\n\nfunction checkAllFieldsSelected(fields): boolean {\n  return fields.every((field) => {\n    if (field.fields) {\n      return checkAllFieldsSelected(field.fields)\n    }\n    return field.isChecked || field.isRequired || field.isReadOnly\n  })\n}\n\nfunction checkAtLeastOneFieldChecked(fields): boolean {\n  return fields.some((field) => {\n    if (field.fields) {\n      return field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n    }\n    return field.isChecked\n  })\n}\n\nconst FieldGroupSection = ({\n  group,\n  index,\n  formik,\n}: FieldGroupSectionProps) => {\n  const allAllFieldsSelected = React.useMemo(\n    () => checkAllFieldsSelected(group.fields),\n    [group.fields]\n  )\n\n  const isAtleastOneFieldChecked = React.useMemo(\n    () => checkAtLeastOneFieldChecked(group.fields),\n    [group.fields]\n  )\n\n  const parentKey = `groups[${index}]`\n\n  const handleChangeAllFields = (e) => {\n    const item = get(formik.values, parentKey)\n    function updateAllFields(fields, checked) {\n      return fields.map((field) => {\n        if (field.fields) {\n          return {\n            ...field,\n            fields: updateAllFields(field.fields, checked),\n          }\n        }\n        if (field.isReadOnly || field.isRequired) return field\n        return {\n          ...field,\n          isChecked: checked,\n        }\n      })\n    }\n\n    formik.setFieldValue(parentKey, {\n      ...item,\n      fields: updateAllFields(item.fields, e.target.checked),\n    })\n  }\n\n  if (group.isHidden) return null\n  return (\n    <>\n      <Grid.Item>\n        <StyledGroupCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <CheckboxInput\n            isIndeterminate={!allAllFieldsSelected && isAtleastOneFieldChecked}\n            isChecked={allAllFieldsSelected}\n            onChange={handleChangeAllFields}\n          >\n            {group.groupLabel}\n          </CheckboxInput>\n        </StyledGroupCard>\n      </Grid.Item>\n      <Grid.Item>\n        <StyledFieldsCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <Spacings.Stack scale=\"s\">\n            <FieldSections\n              fields={group.fields}\n              formik={formik}\n              parentKeys={[parentKey]}\n              isAnySiblingChecked={isAtleastOneFieldChecked}\n            />\n          </Spacings.Stack>\n        </StyledFieldsCard>\n      </Grid.Item>\n    </>\n  )\n}\n\nexport const ExportFieldsSelectionStep = () => {\n  const intl = useIntl()\n  const [fieldSearchTerm, setFieldSearchTerm] = React.useState('')\n  const { projectKey } = useApplicationContext((context) => ({\n    projectKey: context.project?.key,\n  }))\n  const [_, setCachedSelectedGroups] = useStorage(\n    `${projectKey}/${OPERATIONS_EXPORT_RESOURCES_MODAL_SELECTION}`,\n    {}\n  )\n  const { formik, setCurrentStep, resourceType, onClose } =\n    useExportResourcesContext()\n  const resourceTypePlural = resourceTypeToDisplayName({\n    resourceType: resourceType,\n    isPlural: true,\n  })\n\n  React.useEffect(() => {\n    formik.validateForm()\n    const selectedFields = {}\n    function getSelectedFields(field, groupName) {\n      const fieldName = field.name ? `${groupName}.${field.name}` : groupName\n      if (field.fields) {\n        return field.fields.forEach((f) => getSelectedFields(f, fieldName))\n      }\n\n      selectedFields[fieldName] = field.isChecked\n    }\n    formik.values.groups.forEach((group) =>\n      getSelectedFields(group, group.groupLabel)\n    )\n    setCachedSelectedGroups({\n      [resourceType]: selectedFields,\n    })\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [formik.values.groups])\n  // adds `isHidden` property to each field based on search term\n  const filteredGroups = React.useMemo(() => {\n    const updateFields = (fields, parentMatches: boolean) => {\n      return fields.map((field) => {\n        const matches = field.label\n          .toLowerCase()\n          .includes(fieldSearchTerm.toLowerCase())\n        let isHidden = !matches && !parentMatches\n        if (field.fields) {\n          const updatedFields = updateFields(field.fields, matches)\n          isHidden = isHidden && updatedFields.every((f) => f.isHidden)\n          field = {\n            ...field,\n            fields: updatedFields,\n          }\n        }\n        return { ...field, isHidden }\n      })\n    }\n    if (!fieldSearchTerm?.trim()?.length) return formik.values.groups\n\n    return formik.values.groups.map((group) => {\n      const updatedFields = updateFields(group.fields, false)\n      const isHidden = updatedFields.every((field) => field.isHidden)\n      return { ...group, fields: updatedFields, isHidden }\n    })\n  }, [formik.values.groups, fieldSearchTerm])\n\n  const onSearchChange = React.useCallback(\n    (event) => {\n      setFieldSearchTerm(event.target.value)\n    },\n    [setFieldSearchTerm]\n  )\n\n  const hasResults = Boolean(filteredGroups.find((group) => !group.isHidden))\n\n  const onSearchChangeThrottled = React.useMemo(\n    () => throttle(onSearchChange, 300),\n    [onSearchChange]\n  )\n\n  const onSearchReset = () => {\n    setFieldSearchTerm('')\n  }\n\n  return (\n    <FormDialog\n      isOpen\n      title={intl.formatMessage(messages.modalTitle, {\n        resourceType: resourceTypePlural,\n      })}\n      labelPrimary={intl.formatMessage(messages.startExport)}\n      onPrimaryButtonClick={formik.submitForm}\n      isPrimaryButtonDisabled={!formik.isValid}\n      labelSecondary={intl.formatMessage(messages.back)}\n      onSecondaryButtonClick={() => setCurrentStep(1)}\n      onClose={onClose}\n      size={16}\n    >\n      <Spacings.Stack scale=\"xl\">\n        <Spacings.Stack scale=\"m\">\n          <Spacings.Inline justifyContent=\"space-between\">\n            <Text.Body intlMessage={messages.selectFieldsToExport} />\n            <Link\n              isExternal\n              tone=\"secondary\"\n              intlMessage={messages.fieldDefinitions}\n              to={FIELD_DEFINITIONS_URLS[resourceType]}\n            />\n          </Spacings.Inline>\n          <SearchTextInput\n            placeholder={intl.formatMessage(messages.searchForFields)}\n            value={fieldSearchTerm}\n            onChange={onSearchChangeThrottled}\n            onReset={onSearchReset}\n            // TODO: `SearchTextInput` props to be improved. The `onSubmit` callback shouldn't be mandatory\n            onSubmit={() => {}}\n          />\n          {hasResults ? (\n            <StyledGridCard insetScale=\"none\">\n              <Grid\n                gridAutoFlow=\"row\"\n                gridTemplateColumns={`${designTokens.constraint5} auto`}\n              >\n                {filteredGroups.map((group, index) => (\n                  <FieldGroupSection\n                    key={group.groupLabel}\n                    group={group}\n                    index={index}\n                    formik={formik}\n                  />\n                ))}\n              </Grid>\n            </StyledGridCard>\n          ) : (\n            <NoSearchResults />\n          )}\n        </Spacings.Stack>\n      </Spacings.Stack>\n    </FormDialog>\n  )\n}\n"]} */"), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["export-fields-selection-step.tsx"],"names":[],"mappings":"AAkCiE","file":"export-fields-selection-step.tsx","sourcesContent":["import React from 'react'\nimport { useIntl } from 'react-intl'\nimport throttle from 'lodash/throttle'\nimport styled from '@emotion/styled'\nimport { css } from '@emotion/react'\nimport { FormDialog } from '@commercetools-frontend/application-components'\nimport {\n  Card,\n  CheckboxInput,\n  Grid,\n  Link,\n  Spacings,\n  Text,\n  SearchTextInput,\n  designTokens,\n} from '@commercetools-frontend/ui-kit'\nimport get from 'lodash/get'\nimport messages from '../messages'\nimport { useExportResourcesContext } from '../@hooks'\nimport {\n  OPERATIONS_EXPORT_RESOURCES_MODAL_SELECTION,\n  FIELD_DEFINITIONS_URLS,\n} from '../@constants'\nimport { resourceTypeToDisplayName } from '../@utils'\nimport { NoSearchResults } from './no-search-results'\nimport { FieldGroup, FormikField } from '@types'\nimport { useStorage } from '@commercetools-frontend/experimental-components'\nimport { useApplicationContext } from '@commercetools-frontend/application-shell-connectors'\n\nconst StyledGridCard = styled(Card)`\n  max-height: 50vh;\n  overflow: auto;\n`\n\nconst StyledGroupCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-right: ${designTokens.borderWidth1} solid\n    ${designTokens.colorNeutral90};\n  background-color: ${designTokens.colorNeutral98};\n  height: 100%;\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\nconst StyledFieldsCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\ninterface FilteredField extends FormikField {\n  isHidden?: boolean\n}\ninterface FilteredGroups extends FieldGroup {\n  isHidden?: boolean\n  fields: FilteredField[]\n}\n\ntype FieldSectionsProps = {\n  fields?: FilteredField[]\n  parentKeys: string[]\n  formik: any\n  isAnySiblingChecked: boolean\n}\n\nconst FieldSections = ({\n  fields,\n  parentKeys,\n  formik,\n  isAnySiblingChecked,\n}: FieldSectionsProps) => {\n  return (\n    <Spacings.Stack scale=\"s\">\n      {fields?.map((field, index) => {\n        const isAtleastOneChildChecked = () => {\n          function checkAtLeastOneFieldChecked(fields): boolean {\n            return fields?.some((field) => {\n              if (field.fields) {\n                return (\n                  field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n                )\n              }\n              return field.isChecked\n            })\n          }\n\n          return checkAtLeastOneFieldChecked(field.fields)\n        }\n        const isAnyOfTheChildFieldsChecked = isAtleastOneChildChecked()\n        if (field.isHidden) return null\n        return (\n          <React.Fragment key={`${field.name}-${index}`}>\n            <CheckboxInput\n              isChecked={\n                field.isReadOnly\n                  ? isAnySiblingChecked\n                  : isAnyOfTheChildFieldsChecked || field.isChecked\n              }\n              isReadOnly={field.isReadOnly || field.isRequired}\n              onChange={(e) => {\n                const path = `${parentKeys.join('.')}.fields[${index}]`\n                // Update all child fields\n                const updateAllChildFieldsChecked = (\n                  path: string,\n                  checked: boolean\n                ) => {\n                  const item = get(formik.values, path)\n                  if (item && item.fields) {\n                    item.fields.forEach((field, i) => {\n                      const fieldPath = `${path}.fields[${i}]`\n                      if (!field.fields?.length) {\n                        formik.setFieldValue(\n                          `${fieldPath}.isChecked`,\n                          checked,\n                          // pass `false` to avoid revalidating the form with intermediate values\n                          false\n                        )\n                      }\n                      updateAllChildFieldsChecked(fieldPath, checked)\n                    })\n                  }\n                }\n                if (!field.fields?.length) {\n                  formik.setFieldValue(\n                    `${path}.isChecked`,\n                    e.target.checked,\n                    // pass `false` to avoid revalidating the form with intermediate values\n                    false\n                  )\n                }\n                updateAllChildFieldsChecked(path, e.target.checked)\n              }}\n            >\n              {field.label} {field.isRequired && '*'}\n            </CheckboxInput>\n            {field.fields && field.fields?.length > 0 && (\n              <Spacings.Inline scale=\"l\">\n                <div />\n                <FieldSections\n                  fields={field.fields}\n                  formik={formik}\n                  parentKeys={[...parentKeys, `fields[${index}]`]}\n                  isAnySiblingChecked={isAnySiblingChecked}\n                />\n              </Spacings.Inline>\n            )}\n          </React.Fragment>\n        )\n      })}\n    </Spacings.Stack>\n  )\n}\n\ntype FieldGroupSectionProps = {\n  index: number\n  group: FilteredGroups\n  formik: any\n}\n\nfunction checkAllFieldsSelected(fields): boolean {\n  return fields.every((field) => {\n    if (field.fields) {\n      return checkAllFieldsSelected(field.fields)\n    }\n    return field.isChecked || field.isRequired || field.isReadOnly\n  })\n}\n\nfunction checkAtLeastOneFieldChecked(fields): boolean {\n  return fields.some((field) => {\n    if (field.fields) {\n      return field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n    }\n    return field.isChecked\n  })\n}\n\nconst FieldGroupSection = ({\n  group,\n  index,\n  formik,\n}: FieldGroupSectionProps) => {\n  const allAllFieldsSelected = React.useMemo(\n    () => checkAllFieldsSelected(group.fields),\n    [group.fields]\n  )\n\n  const isAtleastOneFieldChecked = React.useMemo(\n    () => checkAtLeastOneFieldChecked(group.fields),\n    [group.fields]\n  )\n\n  const parentKey = `groups[${index}]`\n\n  const handleChangeAllFields = (e) => {\n    const item = get(formik.values, parentKey)\n    function updateAllFields(fields, checked) {\n      return fields.map((field) => {\n        if (field.fields) {\n          return {\n            ...field,\n            fields: updateAllFields(field.fields, checked),\n          }\n        }\n        if (field.isReadOnly || field.isRequired) return field\n        return {\n          ...field,\n          isChecked: checked,\n        }\n      })\n    }\n\n    formik.setFieldValue(parentKey, {\n      ...item,\n      fields: updateAllFields(item.fields, e.target.checked),\n    })\n  }\n\n  if (group.isHidden) return null\n  return (\n    <>\n      <Grid.Item>\n        <StyledGroupCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <CheckboxInput\n            isIndeterminate={!allAllFieldsSelected && isAtleastOneFieldChecked}\n            isChecked={allAllFieldsSelected}\n            onChange={handleChangeAllFields}\n          >\n            {group.groupLabel}\n          </CheckboxInput>\n        </StyledGroupCard>\n      </Grid.Item>\n      <Grid.Item>\n        <StyledFieldsCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <Spacings.Stack scale=\"s\">\n            <FieldSections\n              fields={group.fields}\n              formik={formik}\n              parentKeys={[parentKey]}\n              isAnySiblingChecked={isAtleastOneFieldChecked}\n            />\n          </Spacings.Stack>\n        </StyledFieldsCard>\n      </Grid.Item>\n    </>\n  )\n}\n\nexport const ExportFieldsSelectionStep = () => {\n  const intl = useIntl()\n  const [fieldSearchTerm, setFieldSearchTerm] = React.useState('')\n  const { projectKey } = useApplicationContext((context) => ({\n    projectKey: context.project?.key,\n  }))\n  const [_, setCachedSelectedGroups] = useStorage(\n    `${projectKey}/${OPERATIONS_EXPORT_RESOURCES_MODAL_SELECTION}`,\n    {}\n  )\n  const { formik, setCurrentStep, resourceType, onClose } =\n    useExportResourcesContext()\n  const resourceTypePlural = resourceTypeToDisplayName({\n    resourceType: resourceType,\n    isPlural: true,\n  })\n\n  React.useEffect(() => {\n    formik.validateForm()\n    const selectedFields = {}\n    function getSelectedFields(field, groupName) {\n      const fieldName = field.name ? `${groupName}.${field.name}` : groupName\n      if (field.fields) {\n        return field.fields.forEach((f) => getSelectedFields(f, fieldName))\n      }\n\n      selectedFields[fieldName] = field.isChecked\n    }\n    formik.values.groups.forEach((group) =>\n      getSelectedFields(group, group.groupLabel)\n    )\n    setCachedSelectedGroups({\n      [resourceType]: selectedFields,\n    })\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [formik.values.groups])\n  // adds `isHidden` property to each field based on search term\n  const filteredGroups = React.useMemo(() => {\n    const updateFields = (fields, parentMatches: boolean) => {\n      return fields.map((field) => {\n        const matches = field.label\n          .toLowerCase()\n          .includes(fieldSearchTerm.toLowerCase())\n        let isHidden = !matches && !parentMatches\n        if (field.fields) {\n          const updatedFields = updateFields(field.fields, matches)\n          isHidden = isHidden && updatedFields.every((f) => f.isHidden)\n          field = {\n            ...field,\n            fields: updatedFields,\n          }\n        }\n        return { ...field, isHidden }\n      })\n    }\n    if (!fieldSearchTerm?.trim()?.length) return formik.values.groups\n\n    return formik.values.groups.map((group) => {\n      const updatedFields = updateFields(group.fields, false)\n      const isHidden = updatedFields.every((field) => field.isHidden)\n      return { ...group, fields: updatedFields, isHidden }\n    })\n  }, [formik.values.groups, fieldSearchTerm])\n\n  const onSearchChange = React.useCallback(\n    (event) => {\n      setFieldSearchTerm(event.target.value)\n    },\n    [setFieldSearchTerm]\n  )\n\n  const hasResults = Boolean(filteredGroups.find((group) => !group.isHidden))\n\n  const onSearchChangeThrottled = React.useMemo(\n    () => throttle(onSearchChange, 300),\n    [onSearchChange]\n  )\n\n  const onSearchReset = () => {\n    setFieldSearchTerm('')\n  }\n\n  return (\n    <FormDialog\n      isOpen\n      title={intl.formatMessage(messages.modalTitle, {\n        resourceType: resourceTypePlural,\n      })}\n      labelPrimary={intl.formatMessage(messages.startExport)}\n      onPrimaryButtonClick={formik.submitForm}\n      isPrimaryButtonDisabled={!formik.isValid}\n      labelSecondary={intl.formatMessage(messages.back)}\n      onSecondaryButtonClick={() => setCurrentStep(1)}\n      onClose={onClose}\n      size={16}\n    >\n      <Spacings.Stack scale=\"xl\">\n        <Spacings.Stack scale=\"m\">\n          <Spacings.Inline justifyContent=\"space-between\">\n            <Text.Body intlMessage={messages.selectFieldsToExport} />\n            <Link\n              isExternal\n              tone=\"secondary\"\n              intlMessage={messages.fieldDefinitions}\n              to={FIELD_DEFINITIONS_URLS[resourceType]}\n            />\n          </Spacings.Inline>\n          <SearchTextInput\n            placeholder={intl.formatMessage(messages.searchForFields)}\n            value={fieldSearchTerm}\n            onChange={onSearchChangeThrottled}\n            onReset={onSearchReset}\n            // TODO: `SearchTextInput` props to be improved. The `onSubmit` callback shouldn't be mandatory\n            onSubmit={() => {}}\n          />\n          {hasResults ? (\n            <StyledGridCard insetScale=\"none\">\n              <Grid\n                gridAutoFlow=\"row\"\n                gridTemplateColumns={`${designTokens.constraint5} auto`}\n              >\n                {filteredGroups.map((group, index) => (\n                  <FieldGroupSection\n                    key={group.groupLabel}\n                    group={group}\n                    index={index}\n                    formik={formik}\n                  />\n                ))}\n              </Grid>\n            </StyledGridCard>\n          ) : (\n            <NoSearchResults />\n          )}\n        </Spacings.Stack>\n      </Spacings.Stack>\n    </FormDialog>\n  )\n}\n"]} */"));
|
|
841
867
|
const StyledFieldsCard = /*#__PURE__*/_styled(Card, process.env.NODE_ENV === "production" ? {
|
|
842
868
|
target: "e3xiyhb0"
|
|
843
869
|
} : {
|
|
844
870
|
target: "e3xiyhb0",
|
|
845
871
|
label: "StyledFieldsCard"
|
|
846
|
-
})("border-radius:0;", props => props.showTopBorder && /*#__PURE__*/css("border-top:", designTokens.borderWidth1, " solid ", designTokens.colorNeutral90, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:StyledFieldsCard;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["export-fields-selection-step.tsx"],"names":[],"mappings":"AA+CO","file":"export-fields-selection-step.tsx","sourcesContent":["import React from 'react'\nimport { useIntl } from 'react-intl'\nimport throttle from 'lodash/throttle'\nimport styled from '@emotion/styled'\nimport { css } from '@emotion/react'\nimport { FormDialog } from '@commercetools-frontend/application-components'\nimport {\n  Card,\n  CheckboxInput,\n  Grid,\n  Link,\n  Spacings,\n  Text,\n  SearchTextInput,\n  designTokens,\n} from '@commercetools-frontend/ui-kit'\nimport get from 'lodash/get'\nimport messages from '../messages'\nimport { useExportResourcesContext } from '../@hooks'\nimport { FIELD_DEFINITIONS_URLS } from '../@constants'\nimport { resourceTypeToDisplayName } from '../@utils'\nimport { NoSearchResults } from './no-search-results'\nimport { FieldGroup, FormikField } from '@types'\n\nconst StyledGridCard = styled(Card)`\n  max-height: 50vh;\n  overflow: auto;\n`\n\nconst StyledGroupCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-right: ${designTokens.borderWidth1} solid\n    ${designTokens.colorNeutral90};\n  background-color: ${designTokens.colorNeutral98};\n  height: 100%;\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\nconst StyledFieldsCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\ninterface FilteredField extends FormikField {\n  isHidden?: boolean\n}\ninterface FilteredGroups extends FieldGroup {\n  isHidden?: boolean\n  fields: FilteredField[]\n}\n\ntype FieldSectionsProps = {\n  fields?: FilteredField[]\n  parentKeys: string[]\n  formik: any\n  isAnySiblingChecked: boolean\n}\n\nconst FieldSections = ({\n  fields,\n  parentKeys,\n  formik,\n  isAnySiblingChecked,\n}: FieldSectionsProps) => {\n  return (\n    <Spacings.Stack scale=\"s\">\n      {fields?.map((field, index) => {\n        const isAtleastOneChildChecked = () => {\n          function checkAtLeastOneFieldChecked(fields): boolean {\n            return fields?.some((field) => {\n              if (field.fields) {\n                return (\n                  field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n                )\n              }\n              return field.isChecked\n            })\n          }\n\n          return checkAtLeastOneFieldChecked(field.fields)\n        }\n        const isAnyOfTheChildFieldsChecked = isAtleastOneChildChecked()\n        if (field.isHidden) return null\n        return (\n          <React.Fragment key={`${field.name}-${index}`}>\n            <CheckboxInput\n              isChecked={\n                field.isReadOnly\n                  ? isAnySiblingChecked\n                  : isAnyOfTheChildFieldsChecked || field.isChecked\n              }\n              isReadOnly={field.isReadOnly || field.isRequired}\n              onChange={(e) => {\n                const path = `${parentKeys.join('.')}.fields[${index}]`\n                // Update all child fields\n                const updateAllChildFieldsChecked = (\n                  path: string,\n                  checked: boolean\n                ) => {\n                  const item = get(formik.values, path)\n                  if (item && item.fields) {\n                    item.fields.forEach((field, i) => {\n                      const fieldPath = `${path}.fields[${i}]`\n                      if (!field.fields?.length) {\n                        formik.setFieldValue(\n                          `${fieldPath}.isChecked`,\n                          checked,\n                          // pass `false` to avoid revalidating the form with intermediate values\n                          false\n                        )\n                      }\n                      updateAllChildFieldsChecked(fieldPath, checked)\n                    })\n                  }\n                }\n                if (!field.fields?.length) {\n                  formik.setFieldValue(\n                    `${path}.isChecked`,\n                    e.target.checked,\n                    // pass `false` to avoid revalidating the form with intermediate values\n                    false\n                  )\n                }\n                updateAllChildFieldsChecked(path, e.target.checked)\n              }}\n            >\n              {field.label} {field.isRequired && '*'}\n            </CheckboxInput>\n            {field.fields && field.fields?.length > 0 && (\n              <Spacings.Inline scale=\"l\">\n                <div />\n                <FieldSections\n                  fields={field.fields}\n                  formik={formik}\n                  parentKeys={[...parentKeys, `fields[${index}]`]}\n                  isAnySiblingChecked={isAnySiblingChecked}\n                />\n              </Spacings.Inline>\n            )}\n          </React.Fragment>\n        )\n      })}\n    </Spacings.Stack>\n  )\n}\n\ntype FieldGroupSectionProps = {\n  index: number\n  group: FilteredGroups\n  formik: any\n}\n\nfunction checkAllFieldsSelected(fields): boolean {\n  return fields.every((field) => {\n    if (field.fields) {\n      return checkAllFieldsSelected(field.fields)\n    }\n    return field.isChecked || field.isRequired || field.isReadOnly\n  })\n}\n\nfunction checkAtLeastOneFieldChecked(fields): boolean {\n  return fields.some((field) => {\n    if (field.fields) {\n      return field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n    }\n    return field.isChecked\n  })\n}\n\nconst FieldGroupSection = ({\n  group,\n  index,\n  formik,\n}: FieldGroupSectionProps) => {\n  const allAllFieldsSelected = React.useMemo(\n    () => checkAllFieldsSelected(group.fields),\n    [group.fields]\n  )\n\n  const isAtleastOneFieldChecked = React.useMemo(\n    () => checkAtLeastOneFieldChecked(group.fields),\n    [group.fields]\n  )\n\n  const parentKey = `groups[${index}]`\n\n  const handleChangeAllFields = (e) => {\n    const item = get(formik.values, parentKey)\n    function updateAllFields(fields, checked) {\n      return fields.map((field) => {\n        if (field.fields) {\n          return {\n            ...field,\n            fields: updateAllFields(field.fields, checked),\n          }\n        }\n        if (field.isReadOnly || field.isRequired) return field\n        return {\n          ...field,\n          isChecked: checked,\n        }\n      })\n    }\n\n    formik.setFieldValue(parentKey, {\n      ...item,\n      fields: updateAllFields(item.fields, e.target.checked),\n    })\n  }\n\n  if (group.isHidden) return null\n  return (\n    <>\n      <Grid.Item>\n        <StyledGroupCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <CheckboxInput\n            isIndeterminate={!allAllFieldsSelected && isAtleastOneFieldChecked}\n            isChecked={allAllFieldsSelected}\n            onChange={handleChangeAllFields}\n          >\n            {group.groupLabel}\n          </CheckboxInput>\n        </StyledGroupCard>\n      </Grid.Item>\n      <Grid.Item>\n        <StyledFieldsCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <Spacings.Stack scale=\"s\">\n            <FieldSections\n              fields={group.fields}\n              formik={formik}\n              parentKeys={[parentKey]}\n              isAnySiblingChecked={isAtleastOneFieldChecked}\n            />\n          </Spacings.Stack>\n        </StyledFieldsCard>\n      </Grid.Item>\n    </>\n  )\n}\n\nexport const ExportFieldsSelectionStep = () => {\n  const intl = useIntl()\n  const [fieldSearchTerm, setFieldSearchTerm] = React.useState('')\n  const { formik, setCurrentStep, resourceType, onClose } =\n    useExportResourcesContext()\n  const resourceTypePlural = resourceTypeToDisplayName({\n    resourceType: resourceType,\n    isPlural: true,\n  })\n\n  React.useEffect(() => {\n    formik.validateForm()\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [formik.values.groups])\n  // adds `isHidden` property to each field based on search term\n  const filteredGroups = React.useMemo(() => {\n    const updateFields = (fields, parentMatches: boolean) => {\n      return fields.map((field) => {\n        const matches = field.label\n          .toLowerCase()\n          .includes(fieldSearchTerm.toLowerCase())\n        let isHidden = !matches && !parentMatches\n        if (field.fields) {\n          const updatedFields = updateFields(field.fields, matches)\n          isHidden = isHidden && updatedFields.every((f) => f.isHidden)\n          field = {\n            ...field,\n            fields: updatedFields,\n          }\n        }\n        return { ...field, isHidden }\n      })\n    }\n    if (!fieldSearchTerm?.trim()?.length) return formik.values.groups\n\n    return formik.values.groups.map((group) => {\n      const updatedFields = updateFields(group.fields, false)\n      const isHidden = updatedFields.every((field) => field.isHidden)\n      return { ...group, fields: updatedFields, isHidden }\n    })\n  }, [formik.values.groups, fieldSearchTerm])\n\n  const onSearchChange = React.useCallback(\n    (event) => {\n      setFieldSearchTerm(event.target.value)\n    },\n    [setFieldSearchTerm]\n  )\n\n  const hasResults = Boolean(filteredGroups.find((group) => !group.isHidden))\n\n  const onSearchChangeThrottled = React.useMemo(\n    () => throttle(onSearchChange, 300),\n    [onSearchChange]\n  )\n\n  const onSearchReset = () => {\n    setFieldSearchTerm('')\n  }\n\n  return (\n    <FormDialog\n      isOpen\n      title={intl.formatMessage(messages.modalTitle, {\n        resourceType: resourceTypePlural,\n      })}\n      labelPrimary={intl.formatMessage(messages.startExport)}\n      onPrimaryButtonClick={formik.submitForm}\n      isPrimaryButtonDisabled={!formik.isValid}\n      labelSecondary={intl.formatMessage(messages.back)}\n      onSecondaryButtonClick={() => setCurrentStep(1)}\n      onClose={onClose}\n      size={16}\n    >\n      <Spacings.Stack scale=\"xl\">\n        <Spacings.Stack scale=\"m\">\n          <Spacings.Inline justifyContent=\"space-between\">\n            <Text.Body intlMessage={messages.selectFieldsToExport} />\n            <Link\n              isExternal\n              tone=\"secondary\"\n              intlMessage={messages.fieldDefinitions}\n              to={FIELD_DEFINITIONS_URLS[resourceType]}\n            />\n          </Spacings.Inline>\n          <SearchTextInput\n            placeholder={intl.formatMessage(messages.searchForFields)}\n            value={fieldSearchTerm}\n            onChange={onSearchChangeThrottled}\n            onReset={onSearchReset}\n            // TODO: `SearchTextInput` props to be improved. The `onSubmit` callback shouldn't be mandatory\n            onSubmit={() => {}}\n          />\n          {hasResults ? (\n            <StyledGridCard insetScale=\"none\">\n              <Grid\n                gridAutoFlow=\"row\"\n                gridTemplateColumns={`${designTokens.constraint5} auto`}\n              >\n                {filteredGroups.map((group, index) => (\n                  <FieldGroupSection\n                    key={group.groupLabel}\n                    group={group}\n                    index={index}\n                    formik={formik}\n                  />\n                ))}\n              </Grid>\n            </StyledGridCard>\n          ) : (\n            <NoSearchResults />\n          )}\n        </Spacings.Stack>\n      </Spacings.Stack>\n    </FormDialog>\n  )\n}\n"]} */"), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["export-fields-selection-step.tsx"],"names":[],"mappings":"AA2CkE","file":"export-fields-selection-step.tsx","sourcesContent":["import React from 'react'\nimport { useIntl } from 'react-intl'\nimport throttle from 'lodash/throttle'\nimport styled from '@emotion/styled'\nimport { css } from '@emotion/react'\nimport { FormDialog } from '@commercetools-frontend/application-components'\nimport {\n  Card,\n  CheckboxInput,\n  Grid,\n  Link,\n  Spacings,\n  Text,\n  SearchTextInput,\n  designTokens,\n} from '@commercetools-frontend/ui-kit'\nimport get from 'lodash/get'\nimport messages from '../messages'\nimport { useExportResourcesContext } from '../@hooks'\nimport { FIELD_DEFINITIONS_URLS } from '../@constants'\nimport { resourceTypeToDisplayName } from '../@utils'\nimport { NoSearchResults } from './no-search-results'\nimport { FieldGroup, FormikField } from '@types'\n\nconst StyledGridCard = styled(Card)`\n  max-height: 50vh;\n  overflow: auto;\n`\n\nconst StyledGroupCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-right: ${designTokens.borderWidth1} solid\n    ${designTokens.colorNeutral90};\n  background-color: ${designTokens.colorNeutral98};\n  height: 100%;\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\nconst StyledFieldsCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\ninterface FilteredField extends FormikField {\n  isHidden?: boolean\n}\ninterface FilteredGroups extends FieldGroup {\n  isHidden?: boolean\n  fields: FilteredField[]\n}\n\ntype FieldSectionsProps = {\n  fields?: FilteredField[]\n  parentKeys: string[]\n  formik: any\n  isAnySiblingChecked: boolean\n}\n\nconst FieldSections = ({\n  fields,\n  parentKeys,\n  formik,\n  isAnySiblingChecked,\n}: FieldSectionsProps) => {\n  return (\n    <Spacings.Stack scale=\"s\">\n      {fields?.map((field, index) => {\n        const isAtleastOneChildChecked = () => {\n          function checkAtLeastOneFieldChecked(fields): boolean {\n            return fields?.some((field) => {\n              if (field.fields) {\n                return (\n                  field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n                )\n              }\n              return field.isChecked\n            })\n          }\n\n          return checkAtLeastOneFieldChecked(field.fields)\n        }\n        const isAnyOfTheChildFieldsChecked = isAtleastOneChildChecked()\n        if (field.isHidden) return null\n        return (\n          <React.Fragment key={`${field.name}-${index}`}>\n            <CheckboxInput\n              isChecked={\n                field.isReadOnly\n                  ? isAnySiblingChecked\n                  : isAnyOfTheChildFieldsChecked || field.isChecked\n              }\n              isReadOnly={field.isReadOnly || field.isRequired}\n              onChange={(e) => {\n                const path = `${parentKeys.join('.')}.fields[${index}]`\n                // Update all child fields\n                const updateAllChildFieldsChecked = (\n                  path: string,\n                  checked: boolean\n                ) => {\n                  const item = get(formik.values, path)\n                  if (item && item.fields) {\n                    item.fields.forEach((field, i) => {\n                      const fieldPath = `${path}.fields[${i}]`\n                      if (!field.fields?.length) {\n                        formik.setFieldValue(\n                          `${fieldPath}.isChecked`,\n                          checked,\n                          // pass `false` to avoid revalidating the form with intermediate values\n                          false\n                        )\n                      }\n                      updateAllChildFieldsChecked(fieldPath, checked)\n                    })\n                  }\n                }\n                if (!field.fields?.length) {\n                  formik.setFieldValue(\n                    `${path}.isChecked`,\n                    e.target.checked,\n                    // pass `false` to avoid revalidating the form with intermediate values\n                    false\n                  )\n                }\n                updateAllChildFieldsChecked(path, e.target.checked)\n              }}\n            >\n              {field.label} {field.isRequired && '*'}\n            </CheckboxInput>\n            {field.fields && field.fields?.length > 0 && (\n              <Spacings.Inline scale=\"l\">\n                <div />\n                <FieldSections\n                  fields={field.fields}\n                  formik={formik}\n                  parentKeys={[...parentKeys, `fields[${index}]`]}\n                  isAnySiblingChecked={isAnySiblingChecked}\n                />\n              </Spacings.Inline>\n            )}\n          </React.Fragment>\n        )\n      })}\n    </Spacings.Stack>\n  )\n}\n\ntype FieldGroupSectionProps = {\n  index: number\n  group: FilteredGroups\n  formik: any\n}\n\nfunction checkAllFieldsSelected(fields): boolean {\n  return fields.every((field) => {\n    if (field.fields) {\n      return checkAllFieldsSelected(field.fields)\n    }\n    return field.isChecked || field.isRequired || field.isReadOnly\n  })\n}\n\nfunction checkAtLeastOneFieldChecked(fields): boolean {\n  return fields.some((field) => {\n    if (field.fields) {\n      return field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n    }\n    return field.isChecked\n  })\n}\n\nconst FieldGroupSection = ({\n  group,\n  index,\n  formik,\n}: FieldGroupSectionProps) => {\n  const allAllFieldsSelected = React.useMemo(\n    () => checkAllFieldsSelected(group.fields),\n    [group.fields]\n  )\n\n  const isAtleastOneFieldChecked = React.useMemo(\n    () => checkAtLeastOneFieldChecked(group.fields),\n    [group.fields]\n  )\n\n  const parentKey = `groups[${index}]`\n\n  const handleChangeAllFields = (e) => {\n    const item = get(formik.values, parentKey)\n    function updateAllFields(fields, checked) {\n      return fields.map((field) => {\n        if (field.fields) {\n          return {\n            ...field,\n            fields: updateAllFields(field.fields, checked),\n          }\n        }\n        if (field.isReadOnly || field.isRequired) return field\n        return {\n          ...field,\n          isChecked: checked,\n        }\n      })\n    }\n\n    formik.setFieldValue(parentKey, {\n      ...item,\n      fields: updateAllFields(item.fields, e.target.checked),\n    })\n  }\n\n  if (group.isHidden) return null\n  return (\n    <>\n      <Grid.Item>\n        <StyledGroupCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <CheckboxInput\n            isIndeterminate={!allAllFieldsSelected && isAtleastOneFieldChecked}\n            isChecked={allAllFieldsSelected}\n            onChange={handleChangeAllFields}\n          >\n            {group.groupLabel}\n          </CheckboxInput>\n        </StyledGroupCard>\n      </Grid.Item>\n      <Grid.Item>\n        <StyledFieldsCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <Spacings.Stack scale=\"s\">\n            <FieldSections\n              fields={group.fields}\n              formik={formik}\n              parentKeys={[parentKey]}\n              isAnySiblingChecked={isAtleastOneFieldChecked}\n            />\n          </Spacings.Stack>\n        </StyledFieldsCard>\n      </Grid.Item>\n    </>\n  )\n}\n\nexport const ExportFieldsSelectionStep = () => {\n  const intl = useIntl()\n  const [fieldSearchTerm, setFieldSearchTerm] = React.useState('')\n  const { formik, setCurrentStep, resourceType, onClose } =\n    useExportResourcesContext()\n  const resourceTypePlural = resourceTypeToDisplayName({\n    resourceType: resourceType,\n    isPlural: true,\n  })\n\n  React.useEffect(() => {\n    formik.validateForm()\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [formik.values.groups])\n  // adds `isHidden` property to each field based on search term\n  const filteredGroups = React.useMemo(() => {\n    const updateFields = (fields, parentMatches: boolean) => {\n      return fields.map((field) => {\n        const matches = field.label\n          .toLowerCase()\n          .includes(fieldSearchTerm.toLowerCase())\n        let isHidden = !matches && !parentMatches\n        if (field.fields) {\n          const updatedFields = updateFields(field.fields, matches)\n          isHidden = isHidden && updatedFields.every((f) => f.isHidden)\n          field = {\n            ...field,\n            fields: updatedFields,\n          }\n        }\n        return { ...field, isHidden }\n      })\n    }\n    if (!fieldSearchTerm?.trim()?.length) return formik.values.groups\n\n    return formik.values.groups.map((group) => {\n      const updatedFields = updateFields(group.fields, false)\n      const isHidden = updatedFields.every((field) => field.isHidden)\n      return { ...group, fields: updatedFields, isHidden }\n    })\n  }, [formik.values.groups, fieldSearchTerm])\n\n  const onSearchChange = React.useCallback(\n    (event) => {\n      setFieldSearchTerm(event.target.value)\n    },\n    [setFieldSearchTerm]\n  )\n\n  const hasResults = Boolean(filteredGroups.find((group) => !group.isHidden))\n\n  const onSearchChangeThrottled = React.useMemo(\n    () => throttle(onSearchChange, 300),\n    [onSearchChange]\n  )\n\n  const onSearchReset = () => {\n    setFieldSearchTerm('')\n  }\n\n  return (\n    <FormDialog\n      isOpen\n      title={intl.formatMessage(messages.modalTitle, {\n        resourceType: resourceTypePlural,\n      })}\n      labelPrimary={intl.formatMessage(messages.startExport)}\n      onPrimaryButtonClick={formik.submitForm}\n      isPrimaryButtonDisabled={!formik.isValid}\n      labelSecondary={intl.formatMessage(messages.back)}\n      onSecondaryButtonClick={() => setCurrentStep(1)}\n      onClose={onClose}\n      size={16}\n    >\n      <Spacings.Stack scale=\"xl\">\n        <Spacings.Stack scale=\"m\">\n          <Spacings.Inline justifyContent=\"space-between\">\n            <Text.Body intlMessage={messages.selectFieldsToExport} />\n            <Link\n              isExternal\n              tone=\"secondary\"\n              intlMessage={messages.fieldDefinitions}\n              to={FIELD_DEFINITIONS_URLS[resourceType]}\n            />\n          </Spacings.Inline>\n          <SearchTextInput\n            placeholder={intl.formatMessage(messages.searchForFields)}\n            value={fieldSearchTerm}\n            onChange={onSearchChangeThrottled}\n            onReset={onSearchReset}\n            // TODO: `SearchTextInput` props to be improved. The `onSubmit` callback shouldn't be mandatory\n            onSubmit={() => {}}\n          />\n          {hasResults ? (\n            <StyledGridCard insetScale=\"none\">\n              <Grid\n                gridAutoFlow=\"row\"\n                gridTemplateColumns={`${designTokens.constraint5} auto`}\n              >\n                {filteredGroups.map((group, index) => (\n                  <FieldGroupSection\n                    key={group.groupLabel}\n                    group={group}\n                    index={index}\n                    formik={formik}\n                  />\n                ))}\n              </Grid>\n            </StyledGridCard>\n          ) : (\n            <NoSearchResults />\n          )}\n        </Spacings.Stack>\n      </Spacings.Stack>\n    </FormDialog>\n  )\n}\n"]} */"));
|
|
872
|
+
})("border-radius:0;", props => props.showTopBorder && /*#__PURE__*/css("border-top:", designTokens.borderWidth1, " solid ", designTokens.colorNeutral90, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:StyledFieldsCard;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["export-fields-selection-step.tsx"],"names":[],"mappings":"AAoDO","file":"export-fields-selection-step.tsx","sourcesContent":["import React from 'react'\nimport { useIntl } from 'react-intl'\nimport throttle from 'lodash/throttle'\nimport styled from '@emotion/styled'\nimport { css } from '@emotion/react'\nimport { FormDialog } from '@commercetools-frontend/application-components'\nimport {\n  Card,\n  CheckboxInput,\n  Grid,\n  Link,\n  Spacings,\n  Text,\n  SearchTextInput,\n  designTokens,\n} from '@commercetools-frontend/ui-kit'\nimport get from 'lodash/get'\nimport messages from '../messages'\nimport { useExportResourcesContext } from '../@hooks'\nimport {\n  OPERATIONS_EXPORT_RESOURCES_MODAL_SELECTION,\n  FIELD_DEFINITIONS_URLS,\n} from '../@constants'\nimport { resourceTypeToDisplayName } from '../@utils'\nimport { NoSearchResults } from './no-search-results'\nimport { FieldGroup, FormikField } from '@types'\nimport { useStorage } from '@commercetools-frontend/experimental-components'\nimport { useApplicationContext } from '@commercetools-frontend/application-shell-connectors'\n\nconst StyledGridCard = styled(Card)`\n  max-height: 50vh;\n  overflow: auto;\n`\n\nconst StyledGroupCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-right: ${designTokens.borderWidth1} solid\n    ${designTokens.colorNeutral90};\n  background-color: ${designTokens.colorNeutral98};\n  height: 100%;\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\nconst StyledFieldsCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\ninterface FilteredField extends FormikField {\n  isHidden?: boolean\n}\ninterface FilteredGroups extends FieldGroup {\n  isHidden?: boolean\n  fields: FilteredField[]\n}\n\ntype FieldSectionsProps = {\n  fields?: FilteredField[]\n  parentKeys: string[]\n  formik: any\n  isAnySiblingChecked: boolean\n}\n\nconst FieldSections = ({\n  fields,\n  parentKeys,\n  formik,\n  isAnySiblingChecked,\n}: FieldSectionsProps) => {\n  return (\n    <Spacings.Stack scale=\"s\">\n      {fields?.map((field, index) => {\n        const isAtleastOneChildChecked = () => {\n          function checkAtLeastOneFieldChecked(fields): boolean {\n            return fields?.some((field) => {\n              if (field.fields) {\n                return (\n                  field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n                )\n              }\n              return field.isChecked\n            })\n          }\n\n          return checkAtLeastOneFieldChecked(field.fields)\n        }\n        const isAnyOfTheChildFieldsChecked = isAtleastOneChildChecked()\n        if (field.isHidden) return null\n        return (\n          <React.Fragment key={`${field.name}-${index}`}>\n            <CheckboxInput\n              isChecked={\n                field.isReadOnly\n                  ? isAnySiblingChecked\n                  : isAnyOfTheChildFieldsChecked || field.isChecked\n              }\n              isReadOnly={field.isReadOnly || field.isRequired}\n              onChange={(e) => {\n                const path = `${parentKeys.join('.')}.fields[${index}]`\n                // Update all child fields\n                const updateAllChildFieldsChecked = (\n                  path: string,\n                  checked: boolean\n                ) => {\n                  const item = get(formik.values, path)\n                  if (item && item.fields) {\n                    item.fields.forEach((field, i) => {\n                      const fieldPath = `${path}.fields[${i}]`\n                      if (!field.fields?.length) {\n                        formik.setFieldValue(\n                          `${fieldPath}.isChecked`,\n                          checked,\n                          // pass `false` to avoid revalidating the form with intermediate values\n                          false\n                        )\n                      }\n                      updateAllChildFieldsChecked(fieldPath, checked)\n                    })\n                  }\n                }\n                if (!field.fields?.length) {\n                  formik.setFieldValue(\n                    `${path}.isChecked`,\n                    e.target.checked,\n                    // pass `false` to avoid revalidating the form with intermediate values\n                    false\n                  )\n                }\n                updateAllChildFieldsChecked(path, e.target.checked)\n              }}\n            >\n              {field.label} {field.isRequired && '*'}\n            </CheckboxInput>\n            {field.fields && field.fields?.length > 0 && (\n              <Spacings.Inline scale=\"l\">\n                <div />\n                <FieldSections\n                  fields={field.fields}\n                  formik={formik}\n                  parentKeys={[...parentKeys, `fields[${index}]`]}\n                  isAnySiblingChecked={isAnySiblingChecked}\n                />\n              </Spacings.Inline>\n            )}\n          </React.Fragment>\n        )\n      })}\n    </Spacings.Stack>\n  )\n}\n\ntype FieldGroupSectionProps = {\n  index: number\n  group: FilteredGroups\n  formik: any\n}\n\nfunction checkAllFieldsSelected(fields): boolean {\n  return fields.every((field) => {\n    if (field.fields) {\n      return checkAllFieldsSelected(field.fields)\n    }\n    return field.isChecked || field.isRequired || field.isReadOnly\n  })\n}\n\nfunction checkAtLeastOneFieldChecked(fields): boolean {\n  return fields.some((field) => {\n    if (field.fields) {\n      return field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n    }\n    return field.isChecked\n  })\n}\n\nconst FieldGroupSection = ({\n  group,\n  index,\n  formik,\n}: FieldGroupSectionProps) => {\n  const allAllFieldsSelected = React.useMemo(\n    () => checkAllFieldsSelected(group.fields),\n    [group.fields]\n  )\n\n  const isAtleastOneFieldChecked = React.useMemo(\n    () => checkAtLeastOneFieldChecked(group.fields),\n    [group.fields]\n  )\n\n  const parentKey = `groups[${index}]`\n\n  const handleChangeAllFields = (e) => {\n    const item = get(formik.values, parentKey)\n    function updateAllFields(fields, checked) {\n      return fields.map((field) => {\n        if (field.fields) {\n          return {\n            ...field,\n            fields: updateAllFields(field.fields, checked),\n          }\n        }\n        if (field.isReadOnly || field.isRequired) return field\n        return {\n          ...field,\n          isChecked: checked,\n        }\n      })\n    }\n\n    formik.setFieldValue(parentKey, {\n      ...item,\n      fields: updateAllFields(item.fields, e.target.checked),\n    })\n  }\n\n  if (group.isHidden) return null\n  return (\n    <>\n      <Grid.Item>\n        <StyledGroupCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <CheckboxInput\n            isIndeterminate={!allAllFieldsSelected && isAtleastOneFieldChecked}\n            isChecked={allAllFieldsSelected}\n            onChange={handleChangeAllFields}\n          >\n            {group.groupLabel}\n          </CheckboxInput>\n        </StyledGroupCard>\n      </Grid.Item>\n      <Grid.Item>\n        <StyledFieldsCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <Spacings.Stack scale=\"s\">\n            <FieldSections\n              fields={group.fields}\n              formik={formik}\n              parentKeys={[parentKey]}\n              isAnySiblingChecked={isAtleastOneFieldChecked}\n            />\n          </Spacings.Stack>\n        </StyledFieldsCard>\n      </Grid.Item>\n    </>\n  )\n}\n\nexport const ExportFieldsSelectionStep = () => {\n  const intl = useIntl()\n  const [fieldSearchTerm, setFieldSearchTerm] = React.useState('')\n  const { projectKey } = useApplicationContext((context) => ({\n    projectKey: context.project?.key,\n  }))\n  const [_, setCachedSelectedGroups] = useStorage(\n    `${projectKey}/${OPERATIONS_EXPORT_RESOURCES_MODAL_SELECTION}`,\n    {}\n  )\n  const { formik, setCurrentStep, resourceType, onClose } =\n    useExportResourcesContext()\n  const resourceTypePlural = resourceTypeToDisplayName({\n    resourceType: resourceType,\n    isPlural: true,\n  })\n\n  React.useEffect(() => {\n    formik.validateForm()\n    const selectedFields = {}\n    function getSelectedFields(field, groupName) {\n      const fieldName = field.name ? `${groupName}.${field.name}` : groupName\n      if (field.fields) {\n        return field.fields.forEach((f) => getSelectedFields(f, fieldName))\n      }\n\n      selectedFields[fieldName] = field.isChecked\n    }\n    formik.values.groups.forEach((group) =>\n      getSelectedFields(group, group.groupLabel)\n    )\n    setCachedSelectedGroups({\n      [resourceType]: selectedFields,\n    })\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [formik.values.groups])\n  // adds `isHidden` property to each field based on search term\n  const filteredGroups = React.useMemo(() => {\n    const updateFields = (fields, parentMatches: boolean) => {\n      return fields.map((field) => {\n        const matches = field.label\n          .toLowerCase()\n          .includes(fieldSearchTerm.toLowerCase())\n        let isHidden = !matches && !parentMatches\n        if (field.fields) {\n          const updatedFields = updateFields(field.fields, matches)\n          isHidden = isHidden && updatedFields.every((f) => f.isHidden)\n          field = {\n            ...field,\n            fields: updatedFields,\n          }\n        }\n        return { ...field, isHidden }\n      })\n    }\n    if (!fieldSearchTerm?.trim()?.length) return formik.values.groups\n\n    return formik.values.groups.map((group) => {\n      const updatedFields = updateFields(group.fields, false)\n      const isHidden = updatedFields.every((field) => field.isHidden)\n      return { ...group, fields: updatedFields, isHidden }\n    })\n  }, [formik.values.groups, fieldSearchTerm])\n\n  const onSearchChange = React.useCallback(\n    (event) => {\n      setFieldSearchTerm(event.target.value)\n    },\n    [setFieldSearchTerm]\n  )\n\n  const hasResults = Boolean(filteredGroups.find((group) => !group.isHidden))\n\n  const onSearchChangeThrottled = React.useMemo(\n    () => throttle(onSearchChange, 300),\n    [onSearchChange]\n  )\n\n  const onSearchReset = () => {\n    setFieldSearchTerm('')\n  }\n\n  return (\n    <FormDialog\n      isOpen\n      title={intl.formatMessage(messages.modalTitle, {\n        resourceType: resourceTypePlural,\n      })}\n      labelPrimary={intl.formatMessage(messages.startExport)}\n      onPrimaryButtonClick={formik.submitForm}\n      isPrimaryButtonDisabled={!formik.isValid}\n      labelSecondary={intl.formatMessage(messages.back)}\n      onSecondaryButtonClick={() => setCurrentStep(1)}\n      onClose={onClose}\n      size={16}\n    >\n      <Spacings.Stack scale=\"xl\">\n        <Spacings.Stack scale=\"m\">\n          <Spacings.Inline justifyContent=\"space-between\">\n            <Text.Body intlMessage={messages.selectFieldsToExport} />\n            <Link\n              isExternal\n              tone=\"secondary\"\n              intlMessage={messages.fieldDefinitions}\n              to={FIELD_DEFINITIONS_URLS[resourceType]}\n            />\n          </Spacings.Inline>\n          <SearchTextInput\n            placeholder={intl.formatMessage(messages.searchForFields)}\n            value={fieldSearchTerm}\n            onChange={onSearchChangeThrottled}\n            onReset={onSearchReset}\n            // TODO: `SearchTextInput` props to be improved. The `onSubmit` callback shouldn't be mandatory\n            onSubmit={() => {}}\n          />\n          {hasResults ? (\n            <StyledGridCard insetScale=\"none\">\n              <Grid\n                gridAutoFlow=\"row\"\n                gridTemplateColumns={`${designTokens.constraint5} auto`}\n              >\n                {filteredGroups.map((group, index) => (\n                  <FieldGroupSection\n                    key={group.groupLabel}\n                    group={group}\n                    index={index}\n                    formik={formik}\n                  />\n                ))}\n              </Grid>\n            </StyledGridCard>\n          ) : (\n            <NoSearchResults />\n          )}\n        </Spacings.Stack>\n      </Spacings.Stack>\n    </FormDialog>\n  )\n}\n"]} */"), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["export-fields-selection-step.tsx"],"names":[],"mappings":"AAgDkE","file":"export-fields-selection-step.tsx","sourcesContent":["import React from 'react'\nimport { useIntl } from 'react-intl'\nimport throttle from 'lodash/throttle'\nimport styled from '@emotion/styled'\nimport { css } from '@emotion/react'\nimport { FormDialog } from '@commercetools-frontend/application-components'\nimport {\n  Card,\n  CheckboxInput,\n  Grid,\n  Link,\n  Spacings,\n  Text,\n  SearchTextInput,\n  designTokens,\n} from '@commercetools-frontend/ui-kit'\nimport get from 'lodash/get'\nimport messages from '../messages'\nimport { useExportResourcesContext } from '../@hooks'\nimport {\n  OPERATIONS_EXPORT_RESOURCES_MODAL_SELECTION,\n  FIELD_DEFINITIONS_URLS,\n} from '../@constants'\nimport { resourceTypeToDisplayName } from '../@utils'\nimport { NoSearchResults } from './no-search-results'\nimport { FieldGroup, FormikField } from '@types'\nimport { useStorage } from '@commercetools-frontend/experimental-components'\nimport { useApplicationContext } from '@commercetools-frontend/application-shell-connectors'\n\nconst StyledGridCard = styled(Card)`\n  max-height: 50vh;\n  overflow: auto;\n`\n\nconst StyledGroupCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-right: ${designTokens.borderWidth1} solid\n    ${designTokens.colorNeutral90};\n  background-color: ${designTokens.colorNeutral98};\n  height: 100%;\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\nconst StyledFieldsCard = styled(Card)<{ showTopBorder?: boolean }>`\n  border-radius: 0;\n  ${(props) =>\n    props.showTopBorder &&\n    css`\n      border-top: ${designTokens.borderWidth1} solid\n        ${designTokens.colorNeutral90};\n    `}\n`\n\ninterface FilteredField extends FormikField {\n  isHidden?: boolean\n}\ninterface FilteredGroups extends FieldGroup {\n  isHidden?: boolean\n  fields: FilteredField[]\n}\n\ntype FieldSectionsProps = {\n  fields?: FilteredField[]\n  parentKeys: string[]\n  formik: any\n  isAnySiblingChecked: boolean\n}\n\nconst FieldSections = ({\n  fields,\n  parentKeys,\n  formik,\n  isAnySiblingChecked,\n}: FieldSectionsProps) => {\n  return (\n    <Spacings.Stack scale=\"s\">\n      {fields?.map((field, index) => {\n        const isAtleastOneChildChecked = () => {\n          function checkAtLeastOneFieldChecked(fields): boolean {\n            return fields?.some((field) => {\n              if (field.fields) {\n                return (\n                  field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n                )\n              }\n              return field.isChecked\n            })\n          }\n\n          return checkAtLeastOneFieldChecked(field.fields)\n        }\n        const isAnyOfTheChildFieldsChecked = isAtleastOneChildChecked()\n        if (field.isHidden) return null\n        return (\n          <React.Fragment key={`${field.name}-${index}`}>\n            <CheckboxInput\n              isChecked={\n                field.isReadOnly\n                  ? isAnySiblingChecked\n                  : isAnyOfTheChildFieldsChecked || field.isChecked\n              }\n              isReadOnly={field.isReadOnly || field.isRequired}\n              onChange={(e) => {\n                const path = `${parentKeys.join('.')}.fields[${index}]`\n                // Update all child fields\n                const updateAllChildFieldsChecked = (\n                  path: string,\n                  checked: boolean\n                ) => {\n                  const item = get(formik.values, path)\n                  if (item && item.fields) {\n                    item.fields.forEach((field, i) => {\n                      const fieldPath = `${path}.fields[${i}]`\n                      if (!field.fields?.length) {\n                        formik.setFieldValue(\n                          `${fieldPath}.isChecked`,\n                          checked,\n                          // pass `false` to avoid revalidating the form with intermediate values\n                          false\n                        )\n                      }\n                      updateAllChildFieldsChecked(fieldPath, checked)\n                    })\n                  }\n                }\n                if (!field.fields?.length) {\n                  formik.setFieldValue(\n                    `${path}.isChecked`,\n                    e.target.checked,\n                    // pass `false` to avoid revalidating the form with intermediate values\n                    false\n                  )\n                }\n                updateAllChildFieldsChecked(path, e.target.checked)\n              }}\n            >\n              {field.label} {field.isRequired && '*'}\n            </CheckboxInput>\n            {field.fields && field.fields?.length > 0 && (\n              <Spacings.Inline scale=\"l\">\n                <div />\n                <FieldSections\n                  fields={field.fields}\n                  formik={formik}\n                  parentKeys={[...parentKeys, `fields[${index}]`]}\n                  isAnySiblingChecked={isAnySiblingChecked}\n                />\n              </Spacings.Inline>\n            )}\n          </React.Fragment>\n        )\n      })}\n    </Spacings.Stack>\n  )\n}\n\ntype FieldGroupSectionProps = {\n  index: number\n  group: FilteredGroups\n  formik: any\n}\n\nfunction checkAllFieldsSelected(fields): boolean {\n  return fields.every((field) => {\n    if (field.fields) {\n      return checkAllFieldsSelected(field.fields)\n    }\n    return field.isChecked || field.isRequired || field.isReadOnly\n  })\n}\n\nfunction checkAtLeastOneFieldChecked(fields): boolean {\n  return fields.some((field) => {\n    if (field.fields) {\n      return field.isChecked || checkAtLeastOneFieldChecked(field.fields)\n    }\n    return field.isChecked\n  })\n}\n\nconst FieldGroupSection = ({\n  group,\n  index,\n  formik,\n}: FieldGroupSectionProps) => {\n  const allAllFieldsSelected = React.useMemo(\n    () => checkAllFieldsSelected(group.fields),\n    [group.fields]\n  )\n\n  const isAtleastOneFieldChecked = React.useMemo(\n    () => checkAtLeastOneFieldChecked(group.fields),\n    [group.fields]\n  )\n\n  const parentKey = `groups[${index}]`\n\n  const handleChangeAllFields = (e) => {\n    const item = get(formik.values, parentKey)\n    function updateAllFields(fields, checked) {\n      return fields.map((field) => {\n        if (field.fields) {\n          return {\n            ...field,\n            fields: updateAllFields(field.fields, checked),\n          }\n        }\n        if (field.isReadOnly || field.isRequired) return field\n        return {\n          ...field,\n          isChecked: checked,\n        }\n      })\n    }\n\n    formik.setFieldValue(parentKey, {\n      ...item,\n      fields: updateAllFields(item.fields, e.target.checked),\n    })\n  }\n\n  if (group.isHidden) return null\n  return (\n    <>\n      <Grid.Item>\n        <StyledGroupCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <CheckboxInput\n            isIndeterminate={!allAllFieldsSelected && isAtleastOneFieldChecked}\n            isChecked={allAllFieldsSelected}\n            onChange={handleChangeAllFields}\n          >\n            {group.groupLabel}\n          </CheckboxInput>\n        </StyledGroupCard>\n      </Grid.Item>\n      <Grid.Item>\n        <StyledFieldsCard type=\"flat\" insetScale=\"l\" showTopBorder={true}>\n          <Spacings.Stack scale=\"s\">\n            <FieldSections\n              fields={group.fields}\n              formik={formik}\n              parentKeys={[parentKey]}\n              isAnySiblingChecked={isAtleastOneFieldChecked}\n            />\n          </Spacings.Stack>\n        </StyledFieldsCard>\n      </Grid.Item>\n    </>\n  )\n}\n\nexport const ExportFieldsSelectionStep = () => {\n  const intl = useIntl()\n  const [fieldSearchTerm, setFieldSearchTerm] = React.useState('')\n  const { projectKey } = useApplicationContext((context) => ({\n    projectKey: context.project?.key,\n  }))\n  const [_, setCachedSelectedGroups] = useStorage(\n    `${projectKey}/${OPERATIONS_EXPORT_RESOURCES_MODAL_SELECTION}`,\n    {}\n  )\n  const { formik, setCurrentStep, resourceType, onClose } =\n    useExportResourcesContext()\n  const resourceTypePlural = resourceTypeToDisplayName({\n    resourceType: resourceType,\n    isPlural: true,\n  })\n\n  React.useEffect(() => {\n    formik.validateForm()\n    const selectedFields = {}\n    function getSelectedFields(field, groupName) {\n      const fieldName = field.name ? `${groupName}.${field.name}` : groupName\n      if (field.fields) {\n        return field.fields.forEach((f) => getSelectedFields(f, fieldName))\n      }\n\n      selectedFields[fieldName] = field.isChecked\n    }\n    formik.values.groups.forEach((group) =>\n      getSelectedFields(group, group.groupLabel)\n    )\n    setCachedSelectedGroups({\n      [resourceType]: selectedFields,\n    })\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [formik.values.groups])\n  // adds `isHidden` property to each field based on search term\n  const filteredGroups = React.useMemo(() => {\n    const updateFields = (fields, parentMatches: boolean) => {\n      return fields.map((field) => {\n        const matches = field.label\n          .toLowerCase()\n          .includes(fieldSearchTerm.toLowerCase())\n        let isHidden = !matches && !parentMatches\n        if (field.fields) {\n          const updatedFields = updateFields(field.fields, matches)\n          isHidden = isHidden && updatedFields.every((f) => f.isHidden)\n          field = {\n            ...field,\n            fields: updatedFields,\n          }\n        }\n        return { ...field, isHidden }\n      })\n    }\n    if (!fieldSearchTerm?.trim()?.length) return formik.values.groups\n\n    return formik.values.groups.map((group) => {\n      const updatedFields = updateFields(group.fields, false)\n      const isHidden = updatedFields.every((field) => field.isHidden)\n      return { ...group, fields: updatedFields, isHidden }\n    })\n  }, [formik.values.groups, fieldSearchTerm])\n\n  const onSearchChange = React.useCallback(\n    (event) => {\n      setFieldSearchTerm(event.target.value)\n    },\n    [setFieldSearchTerm]\n  )\n\n  const hasResults = Boolean(filteredGroups.find((group) => !group.isHidden))\n\n  const onSearchChangeThrottled = React.useMemo(\n    () => throttle(onSearchChange, 300),\n    [onSearchChange]\n  )\n\n  const onSearchReset = () => {\n    setFieldSearchTerm('')\n  }\n\n  return (\n    <FormDialog\n      isOpen\n      title={intl.formatMessage(messages.modalTitle, {\n        resourceType: resourceTypePlural,\n      })}\n      labelPrimary={intl.formatMessage(messages.startExport)}\n      onPrimaryButtonClick={formik.submitForm}\n      isPrimaryButtonDisabled={!formik.isValid}\n      labelSecondary={intl.formatMessage(messages.back)}\n      onSecondaryButtonClick={() => setCurrentStep(1)}\n      onClose={onClose}\n      size={16}\n    >\n      <Spacings.Stack scale=\"xl\">\n        <Spacings.Stack scale=\"m\">\n          <Spacings.Inline justifyContent=\"space-between\">\n            <Text.Body intlMessage={messages.selectFieldsToExport} />\n            <Link\n              isExternal\n              tone=\"secondary\"\n              intlMessage={messages.fieldDefinitions}\n              to={FIELD_DEFINITIONS_URLS[resourceType]}\n            />\n          </Spacings.Inline>\n          <SearchTextInput\n            placeholder={intl.formatMessage(messages.searchForFields)}\n            value={fieldSearchTerm}\n            onChange={onSearchChangeThrottled}\n            onReset={onSearchReset}\n            // TODO: `SearchTextInput` props to be improved. The `onSubmit` callback shouldn't be mandatory\n            onSubmit={() => {}}\n          />\n          {hasResults ? (\n            <StyledGridCard insetScale=\"none\">\n              <Grid\n                gridAutoFlow=\"row\"\n                gridTemplateColumns={`${designTokens.constraint5} auto`}\n              >\n                {filteredGroups.map((group, index) => (\n                  <FieldGroupSection\n                    key={group.groupLabel}\n                    group={group}\n                    index={index}\n                    formik={formik}\n                  />\n                ))}\n              </Grid>\n            </StyledGridCard>\n          ) : (\n            <NoSearchResults />\n          )}\n        </Spacings.Stack>\n      </Spacings.Stack>\n    </FormDialog>\n  )\n}\n"]} */"));
|
|
847
873
|
const FieldSections = _ref => {
|
|
848
874
|
let fields = _ref.fields,
|
|
849
875
|
parentKeys = _ref.parentKeys,
|
|
@@ -986,11 +1012,23 @@ const FieldGroupSection = _ref2 => {
|
|
|
986
1012
|
});
|
|
987
1013
|
};
|
|
988
1014
|
const ExportFieldsSelectionStep = () => {
|
|
1015
|
+
var _context5;
|
|
989
1016
|
const intl = useIntl();
|
|
990
1017
|
const _React$useState = React.useState(''),
|
|
991
1018
|
_React$useState2 = _slicedToArray(_React$useState, 2),
|
|
992
1019
|
fieldSearchTerm = _React$useState2[0],
|
|
993
1020
|
setFieldSearchTerm = _React$useState2[1];
|
|
1021
|
+
const _useApplicationContex = useApplicationContext(context => {
|
|
1022
|
+
var _context$project;
|
|
1023
|
+
return {
|
|
1024
|
+
projectKey: (_context$project = context.project) === null || _context$project === void 0 ? void 0 : _context$project.key
|
|
1025
|
+
};
|
|
1026
|
+
}),
|
|
1027
|
+
projectKey = _useApplicationContex.projectKey;
|
|
1028
|
+
const _useStorage = useStorage(_concatInstanceProperty(_context5 = "".concat(projectKey, "/")).call(_context5, OPERATIONS_EXPORT_RESOURCES_MODAL_SELECTION), {}),
|
|
1029
|
+
_useStorage2 = _slicedToArray(_useStorage, 2);
|
|
1030
|
+
_useStorage2[0];
|
|
1031
|
+
const setCachedSelectedGroups = _useStorage2[1];
|
|
994
1032
|
const _useExportResourcesCo = useExportResourcesContext(),
|
|
995
1033
|
formik = _useExportResourcesCo.formik,
|
|
996
1034
|
setCurrentStep = _useExportResourcesCo.setCurrentStep,
|
|
@@ -1001,16 +1039,31 @@ const ExportFieldsSelectionStep = () => {
|
|
|
1001
1039
|
isPlural: true
|
|
1002
1040
|
});
|
|
1003
1041
|
React.useEffect(() => {
|
|
1042
|
+
var _context8;
|
|
1004
1043
|
formik.validateForm();
|
|
1044
|
+
const selectedFields = {};
|
|
1045
|
+
function getSelectedFields(field, groupName) {
|
|
1046
|
+
var _context6;
|
|
1047
|
+
const fieldName = field.name ? _concatInstanceProperty(_context6 = "".concat(groupName, ".")).call(_context6, field.name) : groupName;
|
|
1048
|
+
if (field.fields) {
|
|
1049
|
+
var _context7;
|
|
1050
|
+
return _forEachInstanceProperty(_context7 = field.fields).call(_context7, f => getSelectedFields(f, fieldName));
|
|
1051
|
+
}
|
|
1052
|
+
selectedFields[fieldName] = field.isChecked;
|
|
1053
|
+
}
|
|
1054
|
+
_forEachInstanceProperty(_context8 = _valuesInstanceProperty(formik).groups).call(_context8, group => getSelectedFields(group, group.groupLabel));
|
|
1055
|
+
setCachedSelectedGroups({
|
|
1056
|
+
[resourceType]: selectedFields
|
|
1057
|
+
});
|
|
1005
1058
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
1006
1059
|
}, [_valuesInstanceProperty(formik).groups]);
|
|
1007
1060
|
// adds `isHidden` property to each field based on search term
|
|
1008
1061
|
const filteredGroups = React.useMemo(() => {
|
|
1009
|
-
var _fieldSearchTerm$trim,
|
|
1062
|
+
var _fieldSearchTerm$trim, _context10;
|
|
1010
1063
|
const updateFields = (fields, parentMatches) => {
|
|
1011
1064
|
return _mapInstanceProperty(fields).call(fields, field => {
|
|
1012
|
-
var
|
|
1013
|
-
const matches = _includesInstanceProperty(
|
|
1065
|
+
var _context9;
|
|
1066
|
+
const matches = _includesInstanceProperty(_context9 = field.label.toLowerCase()).call(_context9, fieldSearchTerm.toLowerCase());
|
|
1014
1067
|
let isHidden = !matches && !parentMatches;
|
|
1015
1068
|
if (field.fields) {
|
|
1016
1069
|
const updatedFields = updateFields(field.fields, matches);
|
|
@@ -1025,7 +1078,7 @@ const ExportFieldsSelectionStep = () => {
|
|
|
1025
1078
|
});
|
|
1026
1079
|
};
|
|
1027
1080
|
if (!(fieldSearchTerm !== null && fieldSearchTerm !== void 0 && (_fieldSearchTerm$trim = _trimInstanceProperty(fieldSearchTerm).call(fieldSearchTerm)) !== null && _fieldSearchTerm$trim !== void 0 && _fieldSearchTerm$trim.length)) return _valuesInstanceProperty(formik).groups;
|
|
1028
|
-
return _mapInstanceProperty(
|
|
1081
|
+
return _mapInstanceProperty(_context10 = _valuesInstanceProperty(formik).groups).call(_context10, group => {
|
|
1029
1082
|
const updatedFields = updateFields(group.fields, false);
|
|
1030
1083
|
const isHidden = _everyInstanceProperty(updatedFields).call(updatedFields, field => field.isHidden);
|
|
1031
1084
|
return _objectSpread$1(_objectSpread$1({}, group), {}, {
|