payload
Version:
Node, React and MongoDB Headless CMS and Application Framework
337 lines (336 loc) • 35.4 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", {
value: true
});
Object.defineProperty(exports, "default", {
enumerable: true,
get: function() {
return _default;
}
});
const _react = /*#__PURE__*/ _interop_require_wildcard(require("react"));
const _reacti18next = require("react-i18next");
const _useDebounce = /*#__PURE__*/ _interop_require_default(require("../../../../../hooks/useDebounce"));
const _Config = require("../../../../utilities/Config");
const _ReactSelect = /*#__PURE__*/ _interop_require_default(require("../../../ReactSelect"));
require("./index.scss");
const _optionsReducer = /*#__PURE__*/ _interop_require_default(require("./optionsReducer"));
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 = 'condition-value-relationship';
const maxResultsPerRequest = 10;
const RelationshipField = (props)=>{
const { admin: { isSortable } = {}, disabled, hasMany, onChange, relationTo, value } = props;
const { collections, routes: { api }, serverURL } = (0, _Config.useConfig)();
const hasMultipleRelations = Array.isArray(relationTo);
const [options, dispatchOptions] = (0, _react.useReducer)(_optionsReducer.default, []);
const [lastFullyLoadedRelation, setLastFullyLoadedRelation] = (0, _react.useState)(-1);
const [lastLoadedPage, setLastLoadedPage] = (0, _react.useState)(1);
const [search, setSearch] = (0, _react.useState)('');
const [errorLoading, setErrorLoading] = (0, _react.useState)('');
const [hasLoadedFirstOptions, setHasLoadedFirstOptions] = (0, _react.useState)(false);
const debouncedSearch = (0, _useDebounce.default)(search, 300);
const { i18n, t } = (0, _reacti18next.useTranslation)('general');
const addOptions = (0, _react.useCallback)((data, relation)=>{
const collection = collections.find((coll)=>coll.slug === relation);
dispatchOptions({
collection,
data,
hasMultipleRelations,
i18n,
relation,
type: 'ADD'
});
}, [
collections,
hasMultipleRelations,
i18n
]);
const getResults = (0, _react.useCallback)(async ({ lastFullyLoadedRelation: lastFullyLoadedRelationArg, lastLoadedPage: lastLoadedPageArg, search: searchArg })=>{
let lastLoadedPageToUse = typeof lastLoadedPageArg !== 'undefined' ? lastLoadedPageArg : 1;
const lastFullyLoadedRelationToUse = typeof lastFullyLoadedRelationArg !== 'undefined' ? lastFullyLoadedRelationArg : -1;
const relations = Array.isArray(relationTo) ? relationTo : [
relationTo
];
const relationsToFetch = lastFullyLoadedRelationToUse === -1 ? relations : relations.slice(lastFullyLoadedRelationToUse + 1);
let resultsFetched = 0;
if (!errorLoading) {
relationsToFetch.reduce(async (priorRelation, relation)=>{
await priorRelation;
if (resultsFetched < 10) {
const collection = collections.find((coll)=>coll.slug === relation);
const fieldToSearch = collection?.admin?.useAsTitle || 'id';
const searchParam = searchArg ? `&where[${fieldToSearch}][like]=${searchArg}` : '';
const response = await fetch(`${serverURL}${api}/${relation}?limit=${maxResultsPerRequest}&page=${lastLoadedPageToUse}&depth=0${searchParam}`, {
credentials: 'include',
headers: {
'Accept-Language': i18n.language
}
});
if (response.ok) {
const data = await response.json();
if (data.docs.length > 0) {
resultsFetched += data.docs.length;
addOptions(data, relation);
setLastLoadedPage(data.page);
if (!data.nextPage) {
setLastFullyLoadedRelation(relations.indexOf(relation));
// If there are more relations to search, need to reset lastLoadedPage to 1
// both locally within function and state
if (relations.indexOf(relation) + 1 < relations.length) {
lastLoadedPageToUse = 1;
}
}
}
} else {
setErrorLoading(t('error:unspecific'));
}
}
}, Promise.resolve());
}
}, [
i18n,
relationTo,
errorLoading,
collections,
serverURL,
api,
addOptions,
t
]);
const findOptionsByValue = (0, _react.useCallback)(()=>{
if (value) {
if (hasMany) {
if (Array.isArray(value)) {
return value.map((val)=>{
if (hasMultipleRelations) {
let matchedOption;
options.forEach((opt)=>{
if (opt.options) {
opt.options.some((subOpt)=>{
if (subOpt?.value === val.value) {
matchedOption = subOpt;
return true;
}
return false;
});
}
});
return matchedOption;
}
return options.find((opt)=>opt.value === val);
});
}
return undefined;
}
if (hasMultipleRelations) {
let matchedOption;
const valueWithRelation = value;
options.forEach((opt)=>{
if (opt?.options) {
opt.options.some((subOpt)=>{
if (subOpt?.value === valueWithRelation.value) {
matchedOption = subOpt;
return true;
}
return false;
});
}
});
return matchedOption;
}
return options.find((opt)=>opt.value === value);
}
return undefined;
}, [
hasMany,
hasMultipleRelations,
value,
options
]);
const handleInputChange = (0, _react.useCallback)((newSearch)=>{
if (search !== newSearch) {
setSearch(newSearch);
}
}, [
search
]);
const addOptionByID = (0, _react.useCallback)(async (id, relation)=>{
if (!errorLoading && id !== 'null') {
const response = await fetch(`${serverURL}${api}/${relation}/${id}?depth=0`, {
credentials: 'include',
headers: {
'Accept-Language': i18n.language
}
});
if (response.ok) {
const data = await response.json();
addOptions({
docs: [
data
]
}, relation);
} else {
console.error(t('error:loadingDocument', {
id
}));
}
}
}, [
i18n,
addOptions,
api,
errorLoading,
serverURL,
t
]);
// ///////////////////////////
// Get results when search input changes
// ///////////////////////////
(0, _react.useEffect)(()=>{
dispatchOptions({
i18n,
required: true,
type: 'CLEAR'
});
setHasLoadedFirstOptions(true);
setLastLoadedPage(1);
setLastFullyLoadedRelation(-1);
getResults({
search: debouncedSearch
});
}, [
getResults,
debouncedSearch,
relationTo,
i18n
]);
// ///////////////////////////
// Format options once first options have been retrieved
// ///////////////////////////
(0, _react.useEffect)(()=>{
if (value && hasLoadedFirstOptions) {
if (hasMany) {
const matchedOptions = findOptionsByValue();
(matchedOptions || []).forEach((option, i)=>{
if (!option) {
if (hasMultipleRelations) {
addOptionByID(value[i].value, value[i].relationTo);
} else {
addOptionByID(value[i], relationTo);
}
}
});
} else {
const matchedOption = findOptionsByValue();
if (!matchedOption) {
if (hasMultipleRelations) {
const valueWithRelation = value;
addOptionByID(valueWithRelation.value, valueWithRelation.relationTo);
} else {
addOptionByID(value, relationTo);
}
}
}
}
}, [
addOptionByID,
findOptionsByValue,
hasMany,
hasMultipleRelations,
relationTo,
value,
hasLoadedFirstOptions
]);
const classes = [
'field-type',
baseClass,
errorLoading && 'error-loading'
].filter(Boolean).join(' ');
const valueToRender = findOptionsByValue() || value;
return /*#__PURE__*/ _react.default.createElement("div", {
className: classes
}, !errorLoading && /*#__PURE__*/ _react.default.createElement(_ReactSelect.default, {
disabled: disabled,
isMulti: hasMany,
isSortable: isSortable,
onChange: (selected)=>{
if (hasMany) {
onChange(selected ? selected.map((option)=>{
if (hasMultipleRelations) {
return {
relationTo: option.relationTo,
value: option.value
};
}
return option.value;
}) : null);
} else if (hasMultipleRelations) {
onChange({
relationTo: selected.relationTo,
value: selected.value
});
} else {
onChange(selected.value);
}
},
onInputChange: handleInputChange,
onMenuScrollToBottom: ()=>{
getResults({
lastFullyLoadedRelation,
lastLoadedPage: lastLoadedPage + 1
});
},
options: options,
placeholder: t('selectValue'),
value: valueToRender
}), errorLoading && /*#__PURE__*/ _react.default.createElement("div", {
className: `${baseClass}__error-loading`
}, errorLoading));
};
const _default = RelationshipField;
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../../../../../src/admin/components/elements/WhereBuilder/Condition/Relationship/index.tsx"],"sourcesContent":["import React, { useCallback, useEffect, useReducer, useState } from 'react'\nimport { useTranslation } from 'react-i18next'\n\nimport type { PaginatedDocs } from '../../../../../../database/types'\nimport type { Option } from '../../../ReactSelect/types'\nimport type { GetResults, Props, ValueWithRelation } from './types'\n\nimport useDebounce from '../../../../../hooks/useDebounce'\nimport { useConfig } from '../../../../utilities/Config'\nimport ReactSelect from '../../../ReactSelect'\nimport './index.scss'\nimport optionsReducer from './optionsReducer'\n\nconst baseClass = 'condition-value-relationship'\n\nconst maxResultsPerRequest = 10\n\nconst RelationshipField: React.FC<Props> = (props) => {\n  const { admin: { isSortable } = {}, disabled, hasMany, onChange, relationTo, value } = props\n\n  const {\n    collections,\n    routes: { api },\n    serverURL,\n  } = useConfig()\n\n  const hasMultipleRelations = Array.isArray(relationTo)\n  const [options, dispatchOptions] = useReducer(optionsReducer, [])\n  const [lastFullyLoadedRelation, setLastFullyLoadedRelation] = useState(-1)\n  const [lastLoadedPage, setLastLoadedPage] = useState(1)\n  const [search, setSearch] = useState('')\n  const [errorLoading, setErrorLoading] = useState('')\n  const [hasLoadedFirstOptions, setHasLoadedFirstOptions] = useState(false)\n  const debouncedSearch = useDebounce(search, 300)\n  const { i18n, t } = useTranslation('general')\n\n  const addOptions = useCallback(\n    (data, relation) => {\n      const collection = collections.find((coll) => coll.slug === relation)\n      dispatchOptions({ collection, data, hasMultipleRelations, i18n, relation, type: 'ADD' })\n    },\n    [collections, hasMultipleRelations, i18n],\n  )\n\n  const getResults = useCallback<GetResults>(\n    async ({\n      lastFullyLoadedRelation: lastFullyLoadedRelationArg,\n      lastLoadedPage: lastLoadedPageArg,\n      search: searchArg,\n    }) => {\n      let lastLoadedPageToUse = typeof lastLoadedPageArg !== 'undefined' ? lastLoadedPageArg : 1\n      const lastFullyLoadedRelationToUse =\n        typeof lastFullyLoadedRelationArg !== 'undefined' ? lastFullyLoadedRelationArg : -1\n\n      const relations = Array.isArray(relationTo) ? relationTo : [relationTo]\n      const relationsToFetch =\n        lastFullyLoadedRelationToUse === -1\n          ? relations\n          : relations.slice(lastFullyLoadedRelationToUse + 1)\n\n      let resultsFetched = 0\n\n      if (!errorLoading) {\n        relationsToFetch.reduce(async (priorRelation, relation) => {\n          await priorRelation\n\n          if (resultsFetched < 10) {\n            const collection = collections.find((coll) => coll.slug === relation)\n            const fieldToSearch = collection?.admin?.useAsTitle || 'id'\n            const searchParam = searchArg ? `&where[${fieldToSearch}][like]=${searchArg}` : ''\n\n            const response = await fetch(\n              `${serverURL}${api}/${relation}?limit=${maxResultsPerRequest}&page=${lastLoadedPageToUse}&depth=0${searchParam}`,\n              {\n                credentials: 'include',\n                headers: {\n                  'Accept-Language': i18n.language,\n                },\n              },\n            )\n\n            if (response.ok) {\n              const data: PaginatedDocs = await response.json()\n              if (data.docs.length > 0) {\n                resultsFetched += data.docs.length\n                addOptions(data, relation)\n                setLastLoadedPage(data.page)\n\n                if (!data.nextPage) {\n                  setLastFullyLoadedRelation(relations.indexOf(relation))\n\n                  // If there are more relations to search, need to reset lastLoadedPage to 1\n                  // both locally within function and state\n                  if (relations.indexOf(relation) + 1 < relations.length) {\n                    lastLoadedPageToUse = 1\n                  }\n                }\n              }\n            } else {\n              setErrorLoading(t('error:unspecific'))\n            }\n          }\n        }, Promise.resolve())\n      }\n    },\n    [i18n, relationTo, errorLoading, collections, serverURL, api, addOptions, t],\n  )\n\n  const findOptionsByValue = useCallback((): Option | Option[] => {\n    if (value) {\n      if (hasMany) {\n        if (Array.isArray(value)) {\n          return value.map((val) => {\n            if (hasMultipleRelations) {\n              let matchedOption: Option\n\n              options.forEach((opt) => {\n                if (opt.options) {\n                  opt.options.some((subOpt) => {\n                    if (subOpt?.value === val.value) {\n                      matchedOption = subOpt\n                      return true\n                    }\n\n                    return false\n                  })\n                }\n              })\n\n              return matchedOption\n            }\n\n            return options.find((opt) => opt.value === val)\n          })\n        }\n\n        return undefined\n      }\n\n      if (hasMultipleRelations) {\n        let matchedOption: Option\n\n        const valueWithRelation = value as ValueWithRelation\n\n        options.forEach((opt) => {\n          if (opt?.options) {\n            opt.options.some((subOpt) => {\n              if (subOpt?.value === valueWithRelation.value) {\n                matchedOption = subOpt\n                return true\n              }\n              return false\n            })\n          }\n        })\n\n        return matchedOption\n      }\n\n      return options.find((opt) => opt.value === value)\n    }\n\n    return undefined\n  }, [hasMany, hasMultipleRelations, value, options])\n\n  const handleInputChange = useCallback(\n    (newSearch) => {\n      if (search !== newSearch) {\n        setSearch(newSearch)\n      }\n    },\n    [search],\n  )\n\n  const addOptionByID = useCallback(\n    async (id, relation) => {\n      if (!errorLoading && id !== 'null') {\n        const response = await fetch(`${serverURL}${api}/${relation}/${id}?depth=0`, {\n          credentials: 'include',\n          headers: {\n            'Accept-Language': i18n.language,\n          },\n        })\n\n        if (response.ok) {\n          const data = await response.json()\n          addOptions({ docs: [data] }, relation)\n        } else {\n          console.error(t('error:loadingDocument', { id }))\n        }\n      }\n    },\n    [i18n, addOptions, api, errorLoading, serverURL, t],\n  )\n\n  // ///////////////////////////\n  // Get results when search input changes\n  // ///////////////////////////\n\n  useEffect(() => {\n    dispatchOptions({\n      i18n,\n      required: true,\n      type: 'CLEAR',\n    })\n\n    setHasLoadedFirstOptions(true)\n    setLastLoadedPage(1)\n    setLastFullyLoadedRelation(-1)\n    getResults({ search: debouncedSearch })\n  }, [getResults, debouncedSearch, relationTo, i18n])\n\n  // ///////////////////////////\n  // Format options once first options have been retrieved\n  // ///////////////////////////\n\n  useEffect(() => {\n    if (value && hasLoadedFirstOptions) {\n      if (hasMany) {\n        const matchedOptions = findOptionsByValue()\n\n        ;((matchedOptions as Option[]) || []).forEach((option, i) => {\n          if (!option) {\n            if (hasMultipleRelations) {\n              addOptionByID(value[i].value, value[i].relationTo)\n            } else {\n              addOptionByID(value[i], relationTo)\n            }\n          }\n        })\n      } else {\n        const matchedOption = findOptionsByValue()\n\n        if (!matchedOption) {\n          if (hasMultipleRelations) {\n            const valueWithRelation = value as ValueWithRelation\n            addOptionByID(valueWithRelation.value, valueWithRelation.relationTo)\n          } else {\n            addOptionByID(value, relationTo)\n          }\n        }\n      }\n    }\n  }, [\n    addOptionByID,\n    findOptionsByValue,\n    hasMany,\n    hasMultipleRelations,\n    relationTo,\n    value,\n    hasLoadedFirstOptions,\n  ])\n\n  const classes = ['field-type', baseClass, errorLoading && 'error-loading']\n    .filter(Boolean)\n    .join(' ')\n\n  const valueToRender = (findOptionsByValue() || value) as Option\n\n  return (\n    <div className={classes}>\n      {!errorLoading && (\n        <ReactSelect\n          disabled={disabled}\n          isMulti={hasMany}\n          isSortable={isSortable}\n          onChange={(selected) => {\n            if (hasMany) {\n              onChange(\n                selected\n                  ? selected.map((option) => {\n                      if (hasMultipleRelations) {\n                        return {\n                          relationTo: option.relationTo,\n                          value: option.value,\n                        }\n                      }\n\n                      return option.value\n                    })\n                  : null,\n              )\n            } else if (hasMultipleRelations) {\n              onChange({\n                relationTo: selected.relationTo,\n                value: selected.value,\n              })\n            } else {\n              onChange(selected.value)\n            }\n          }}\n          onInputChange={handleInputChange}\n          onMenuScrollToBottom={() => {\n            getResults({ lastFullyLoadedRelation, lastLoadedPage: lastLoadedPage + 1 })\n          }}\n          options={options}\n          placeholder={t('selectValue')}\n          value={valueToRender}\n        />\n      )}\n      {errorLoading && <div className={`${baseClass}__error-loading`}>{errorLoading}</div>}\n    </div>\n  )\n}\n\nexport default RelationshipField\n"],"names":["baseClass","maxResultsPerRequest","RelationshipField","props","admin","isSortable","disabled","hasMany","onChange","relationTo","value","collections","routes","api","serverURL","useConfig","hasMultipleRelations","Array","isArray","options","dispatchOptions","useReducer","optionsReducer","lastFullyLoadedRelation","setLastFullyLoadedRelation","useState","lastLoadedPage","setLastLoadedPage","search","setSearch","errorLoading","setErrorLoading","hasLoadedFirstOptions","setHasLoadedFirstOptions","debouncedSearch","useDebounce","i18n","t","useTranslation","addOptions","useCallback","data","relation","collection","find","coll","slug","type","getResults","lastFullyLoadedRelationArg","lastLoadedPageArg","searchArg","lastLoadedPageToUse","lastFullyLoadedRelationToUse","relations","relationsToFetch","slice","resultsFetched","reduce","priorRelation","fieldToSearch","useAsTitle","searchParam","response","fetch","credentials","headers","language","ok","json","docs","length","page","nextPage","indexOf","Promise","resolve","findOptionsByValue","map","val","matchedOption","forEach","opt","some","subOpt","undefined","valueWithRelation","handleInputChange","newSearch","addOptionByID","id","console","error","useEffect","required","matchedOptions","option","i","classes","filter","Boolean","join","valueToRender","div","className","ReactSelect","isMulti","selected","onInputChange","onMenuScrollToBottom","placeholder"],"mappings":"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"}