UNPKG

payload

Version:

Node, React and MongoDB Headless CMS and Application Framework

232 lines (231 loc) • 27.3 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); Object.defineProperty(exports, "default", { enumerable: true, get: function() { return _default; } }); const _qs = /*#__PURE__*/ _interop_require_default(require("qs")); const _react = /*#__PURE__*/ _interop_require_wildcard(require("react")); const _reacti18next = require("react-i18next"); const _reactrouterdom = require("react-router-dom"); const _flattenTopLevelFields = /*#__PURE__*/ _interop_require_default(require("../../../../utilities/flattenTopLevelFields")); const _getTranslation = require("../../../../utilities/getTranslation"); const _useThrottledEffect = /*#__PURE__*/ _interop_require_default(require("../../../hooks/useThrottledEffect")); const _SearchParams = require("../../utilities/SearchParams"); const _Button = /*#__PURE__*/ _interop_require_default(require("../Button")); const _Condition = /*#__PURE__*/ _interop_require_default(require("./Condition")); const _fieldtypes = /*#__PURE__*/ _interop_require_default(require("./field-types")); require("./index.scss"); const _reducer = /*#__PURE__*/ _interop_require_default(require("./reducer")); const _transformWhereQuery = require("./transformWhereQuery"); const _validateWhereQuery = /*#__PURE__*/ _interop_require_default(require("./validateWhereQuery")); function _interop_require_default(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _getRequireWildcardCache(nodeInterop) { if (typeof WeakMap !== "function") return null; var cacheBabelInterop = new WeakMap(); var cacheNodeInterop = new WeakMap(); return (_getRequireWildcardCache = function(nodeInterop) { return nodeInterop ? cacheNodeInterop : cacheBabelInterop; })(nodeInterop); } function _interop_require_wildcard(obj, nodeInterop) { if (!nodeInterop && obj && obj.__esModule) { return obj; } if (obj === null || typeof obj !== "object" && typeof obj !== "function") { return { default: obj }; } var cache = _getRequireWildcardCache(nodeInterop); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = { __proto__: null }; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for(var key in obj){ if (key !== "default" && Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj.default = obj; if (cache) { cache.set(obj, newObj); } return newObj; } const baseClass = 'where-builder'; const reduceFields = (fields, i18n)=>(0, _flattenTopLevelFields.default)(fields).reduce((reduced, field)=>{ if (typeof _fieldtypes.default[field.type] === 'object') { const operatorKeys = new Set(); const operators = _fieldtypes.default[field.type].operators.reduce((acc, operator)=>{ if (!operatorKeys.has(operator.value)) { operatorKeys.add(operator.value); return [ ...acc, { ...operator, label: i18n.t(`operators:${operator.label}`) } ]; } return acc; }, []); const formattedField = { label: (0, _getTranslation.getTranslation)(field.label || field.name, i18n), value: field.name, ..._fieldtypes.default[field.type], operators, props: { ...field } }; return [ ...reduced, formattedField ]; } return reduced; }, []); /** * The WhereBuilder component is used to render the filter controls for a collection's list view. * It is part of the {@link ListControls} component which is used to render the controls (search, filter, where). */ const WhereBuilder = (props)=>{ const { collection: { labels: { plural } = {} } = {}, collection, handleChange, modifySearchQuery = true } = props; const history = (0, _reactrouterdom.useHistory)(); const params = (0, _SearchParams.useSearchParams)(); const { i18n, t } = (0, _reacti18next.useTranslation)('general'); // This handles initializing the where conditions from the search query (URL). That way, if you pass in // query params to the URL, the where conditions will be initialized from those and displayed in the UI. // Example: /admin/collections/posts?where[or][0][and][0][text][equals]=example%20post const [conditions, dispatchConditions] = (0, _react.useReducer)(_reducer.default, params.where, (whereFromSearch)=>{ if (modifySearchQuery && whereFromSearch) { if ((0, _validateWhereQuery.default)(whereFromSearch)) { return whereFromSearch.or; } // Transform the where query to be in the right format. This will transform something simple like [text][equals]=example%20post to the right format const transformedWhere = (0, _transformWhereQuery.transformWhereQuery)(whereFromSearch); if ((0, _validateWhereQuery.default)(transformedWhere)) { return transformedWhere.or; } console.warn('Invalid where query in URL. Ignoring.'); } return []; }); const [reducedFields] = (0, _react.useState)(()=>reduceFields(collection.fields, i18n)); // This handles updating the search query (URL) when the where conditions change (0, _useThrottledEffect.default)(()=>{ const currentParams = _qs.default.parse(history.location.search, { depth: 10, ignoreQueryPrefix: true }); const paramsToKeep = typeof currentParams?.where === 'object' && 'or' in currentParams.where ? currentParams.where.or.reduce((keptParams, param)=>{ const newParam = { ...param }; if (param.and) { delete newParam.and; } return [ ...keptParams, newParam ]; }, []) : []; const hasNewWhereConditions = conditions.length > 0; const newWhereQuery = { ...typeof currentParams?.where === 'object' && ((0, _validateWhereQuery.default)(currentParams?.where) || !hasNewWhereConditions) ? currentParams.where : {}, or: [ ...conditions, ...paramsToKeep ] }; if (handleChange) handleChange(newWhereQuery); const hasExistingConditions = typeof currentParams?.where === 'object' && 'or' in currentParams.where; if (modifySearchQuery && (hasExistingConditions && !hasNewWhereConditions || hasNewWhereConditions)) { history.replace({ search: _qs.default.stringify({ ...currentParams, page: 1, where: newWhereQuery }, { addQueryPrefix: true }) }); } }, 500, [ conditions, modifySearchQuery, handleChange ]); return /*#__PURE__*/ _react.default.createElement("div", { className: baseClass }, conditions.length > 0 && /*#__PURE__*/ _react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/ _react.default.createElement("div", { className: `${baseClass}__label` }, t('filterWhere', { label: (0, _getTranslation.getTranslation)(plural, i18n) })), /*#__PURE__*/ _react.default.createElement("ul", { className: `${baseClass}__or-filters` }, conditions.map((or, orIndex)=>/*#__PURE__*/ _react.default.createElement("li", { key: orIndex }, orIndex !== 0 && /*#__PURE__*/ _react.default.createElement("div", { className: `${baseClass}__label` }, t('or')), /*#__PURE__*/ _react.default.createElement("ul", { className: `${baseClass}__and-filters` }, Array.isArray(or?.and) && or.and.map((_, andIndex)=>/*#__PURE__*/ _react.default.createElement("li", { key: andIndex }, andIndex !== 0 && /*#__PURE__*/ _react.default.createElement("div", { className: `${baseClass}__label` }, t('and')), /*#__PURE__*/ _react.default.createElement(_Condition.default, { andIndex: andIndex, dispatch: dispatchConditions, fields: reducedFields, key: andIndex, orIndex: orIndex, value: conditions[orIndex].and[andIndex] }))))))), /*#__PURE__*/ _react.default.createElement(_Button.default, { buttonStyle: "icon-label", className: `${baseClass}__add-or`, icon: "plus", iconPosition: "left", iconStyle: "with-border", onClick: ()=>{ if (reducedFields.length > 0) dispatchConditions({ field: reducedFields[0].value, type: 'add' }); } }, t('or'))), conditions.length === 0 && /*#__PURE__*/ _react.default.createElement("div", { className: `${baseClass}__no-filters` }, /*#__PURE__*/ _react.default.createElement("div", { className: `${baseClass}__label` }, t('noFiltersSet')), /*#__PURE__*/ _react.default.createElement(_Button.default, { buttonStyle: "icon-label", className: `${baseClass}__add-first-filter`, icon: "plus", iconPosition: "left", iconStyle: "with-border", onClick: ()=>{ if (reducedFields.length > 0) dispatchConditions({ field: reducedFields[0].value, type: 'add' }); } }, t('addFilter')))); }; const _default = WhereBuilder; //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../../../src/admin/components/elements/WhereBuilder/index.tsx"],"sourcesContent":["import queryString from 'qs'\nimport React, { useReducer, useState } from 'react'\nimport { useTranslation } from 'react-i18next'\nimport { useHistory } from 'react-router-dom'\n\nimport type { Where } from '../../../../types'\nimport type { Props } from './types'\n\nimport flattenTopLevelFields from '../../../../utilities/flattenTopLevelFields'\nimport { getTranslation } from '../../../../utilities/getTranslation'\nimport useThrottledEffect from '../../../hooks/useThrottledEffect'\nimport { useSearchParams } from '../../utilities/SearchParams'\nimport Button from '../Button'\nimport Condition from './Condition'\nimport fieldTypes from './field-types'\nimport './index.scss'\nimport reducer from './reducer'\nimport { transformWhereQuery } from './transformWhereQuery'\nimport validateWhereQuery from './validateWhereQuery'\n\nconst baseClass = 'where-builder'\n\nconst reduceFields = (fields, i18n) =>\n  flattenTopLevelFields(fields).reduce((reduced, field) => {\n    if (typeof fieldTypes[field.type] === 'object') {\n      const operatorKeys = new Set()\n      const operators = fieldTypes[field.type].operators.reduce((acc, operator) => {\n        if (!operatorKeys.has(operator.value)) {\n          operatorKeys.add(operator.value)\n          return [\n            ...acc,\n            {\n              ...operator,\n              label: i18n.t(`operators:${operator.label}`),\n            },\n          ]\n        }\n        return acc\n      }, [])\n\n      const formattedField = {\n        label: getTranslation(field.label || field.name, i18n),\n        value: field.name,\n        ...fieldTypes[field.type],\n        operators,\n        props: {\n          ...field,\n        },\n      }\n\n      return [...reduced, formattedField]\n    }\n\n    return reduced\n  }, [])\n\n/**\n * The WhereBuilder component is used to render the filter controls for a collection's list view.\n * It is part of the {@link ListControls} component which is used to render the controls (search, filter, where).\n */\nconst WhereBuilder: React.FC<Props> = (props) => {\n  const {\n    collection: { labels: { plural } = {} } = {},\n    collection,\n    handleChange,\n    modifySearchQuery = true,\n  } = props\n\n  const history = useHistory()\n  const params = useSearchParams()\n  const { i18n, t } = useTranslation('general')\n\n  // This handles initializing the where conditions from the search query (URL). That way, if you pass in\n  // query params to the URL, the where conditions will be initialized from those and displayed in the UI.\n  // Example: /admin/collections/posts?where[or][0][and][0][text][equals]=example%20post\n  const [conditions, dispatchConditions] = useReducer(reducer, params.where, (whereFromSearch) => {\n    if (modifySearchQuery && whereFromSearch) {\n      if (validateWhereQuery(whereFromSearch)) {\n        return whereFromSearch.or\n      }\n\n      // Transform the where query to be in the right format. This will transform something simple like [text][equals]=example%20post to the right format\n      const transformedWhere = transformWhereQuery(whereFromSearch)\n\n      if (validateWhereQuery(transformedWhere)) {\n        return transformedWhere.or\n      }\n\n      console.warn('Invalid where query in URL. Ignoring.')\n    }\n    return []\n  })\n\n  const [reducedFields] = useState(() => reduceFields(collection.fields, i18n))\n\n  // This handles updating the search query (URL) when the where conditions change\n  useThrottledEffect(\n    () => {\n      const currentParams = queryString.parse(history.location.search, {\n        depth: 10,\n        ignoreQueryPrefix: true,\n      }) as { where: Where }\n\n      const paramsToKeep =\n        typeof currentParams?.where === 'object' && 'or' in currentParams.where\n          ? currentParams.where.or.reduce((keptParams, param) => {\n              const newParam = { ...param }\n              if (param.and) {\n                delete newParam.and\n              }\n              return [...keptParams, newParam]\n            }, [])\n          : []\n\n      const hasNewWhereConditions = conditions.length > 0\n\n      const newWhereQuery = {\n        ...(typeof currentParams?.where === 'object' &&\n        (validateWhereQuery(currentParams?.where) || !hasNewWhereConditions)\n          ? currentParams.where\n          : {}),\n        or: [...conditions, ...paramsToKeep],\n      }\n\n      if (handleChange) handleChange(newWhereQuery as Where)\n\n      const hasExistingConditions =\n        typeof currentParams?.where === 'object' && 'or' in currentParams.where\n\n      if (\n        modifySearchQuery &&\n        ((hasExistingConditions && !hasNewWhereConditions) || hasNewWhereConditions)\n      ) {\n        history.replace({\n          search: queryString.stringify(\n            {\n              ...currentParams,\n              page: 1,\n              where: newWhereQuery,\n            },\n            { addQueryPrefix: true },\n          ),\n        })\n      }\n    },\n    500,\n    [conditions, modifySearchQuery, handleChange],\n  )\n\n  return (\n    <div className={baseClass}>\n      {conditions.length > 0 && (\n        <React.Fragment>\n          <div className={`${baseClass}__label`}>\n            {t('filterWhere', { label: getTranslation(plural, i18n) })}\n          </div>\n          <ul className={`${baseClass}__or-filters`}>\n            {conditions.map((or, orIndex) => (\n              <li key={orIndex}>\n                {orIndex !== 0 && <div className={`${baseClass}__label`}>{t('or')}</div>}\n                <ul className={`${baseClass}__and-filters`}>\n                  {Array.isArray(or?.and) &&\n                    or.and.map((_, andIndex) => (\n                      <li key={andIndex}>\n                        {andIndex !== 0 && <div className={`${baseClass}__label`}>{t('and')}</div>}\n                        <Condition\n                          andIndex={andIndex}\n                          dispatch={dispatchConditions}\n                          fields={reducedFields}\n                          key={andIndex}\n                          orIndex={orIndex}\n                          value={conditions[orIndex].and[andIndex]}\n                        />\n                      </li>\n                    ))}\n                </ul>\n              </li>\n            ))}\n          </ul>\n          <Button\n            buttonStyle=\"icon-label\"\n            className={`${baseClass}__add-or`}\n            icon=\"plus\"\n            iconPosition=\"left\"\n            iconStyle=\"with-border\"\n            onClick={() => {\n              if (reducedFields.length > 0)\n                dispatchConditions({ field: reducedFields[0].value, type: 'add' })\n            }}\n          >\n            {t('or')}\n          </Button>\n        </React.Fragment>\n      )}\n      {conditions.length === 0 && (\n        <div className={`${baseClass}__no-filters`}>\n          <div className={`${baseClass}__label`}>{t('noFiltersSet')}</div>\n          <Button\n            buttonStyle=\"icon-label\"\n            className={`${baseClass}__add-first-filter`}\n            icon=\"plus\"\n            iconPosition=\"left\"\n            iconStyle=\"with-border\"\n            onClick={() => {\n              if (reducedFields.length > 0)\n                dispatchConditions({ field: reducedFields[0].value, type: 'add' })\n            }}\n          >\n            {t('addFilter')}\n          </Button>\n        </div>\n      )}\n    </div>\n  )\n}\n\nexport default WhereBuilder\n"],"names":["baseClass","reduceFields","fields","i18n","flattenTopLevelFields","reduce","reduced","field","fieldTypes","type","operatorKeys","Set","operators","acc","operator","has","value","add","label","t","formattedField","getTranslation","name","props","WhereBuilder","collection","labels","plural","handleChange","modifySearchQuery","history","useHistory","params","useSearchParams","useTranslation","conditions","dispatchConditions","useReducer","reducer","where","whereFromSearch","validateWhereQuery","or","transformedWhere","transformWhereQuery","console","warn","reducedFields","useState","useThrottledEffect","currentParams","queryString","parse","location","search","depth","ignoreQueryPrefix","paramsToKeep","keptParams","param","newParam","and","hasNewWhereConditions","length","newWhereQuery","hasExistingConditions","replace","stringify","page","addQueryPrefix","div","className","React","Fragment","ul","map","orIndex","li","key","Array","isArray","_","andIndex","Condition","dispatch","Button","buttonStyle","icon","iconPosition","iconStyle","onClick"],"mappings":"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"}