payload
Version:
Node, React and MongoDB Headless CMS and Application Framework
309 lines (308 loc) • 28.6 kB
JavaScript
"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 _uuid = require("uuid");
const _usePayloadAPI = /*#__PURE__*/ _interop_require_default(require("../../../../hooks/usePayloadAPI"));
const _useUseAsTitle = require("../../../../hooks/useUseAsTitle");
const _StepNav = require("../../../elements/StepNav");
const _TableColumns = require("../../../elements/TableColumns");
const _ActionsProvider = require("../../../utilities/ActionsProvider");
const _Auth = require("../../../utilities/Auth");
const _Config = require("../../../utilities/Config");
const _Preferences = require("../../../utilities/Preferences");
const _RenderCustomComponent = /*#__PURE__*/ _interop_require_default(require("../../../utilities/RenderCustomComponent"));
const _SearchParams = require("../../../utilities/SearchParams");
const _Default = /*#__PURE__*/ _interop_require_default(require("./Default"));
const _formatFields = /*#__PURE__*/ _interop_require_default(require("./formatFields"));
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 hoistQueryParamsToAnd = (where, queryParams)=>{
if ('and' in where) {
where.and.push(queryParams);
} else if ('or' in where) {
where = {
and: [
where,
queryParams
]
};
} else {
where = {
and: [
where,
queryParams
]
};
}
return where;
};
/**
* The ListView component is table which lists the collection's documents.
* The default list view can be found at the {@link DefaultList} component.
* Users can also create pass their own custom list view component instead
* of using the default one.
*/ const ListView = (props)=>{
const { collection, collection: { admin: { components: { views: { List: CustomList } = {} } = {}, listSearchableFields, pagination: { defaultLimit } }, labels: { plural }, slug } } = props;
const { routes: { admin, api }, serverURL } = (0, _Config.useConfig)();
const { setViewActions } = (0, _ActionsProvider.useActions)();
const preferenceKey = `${collection.slug}-list`;
const { permissions } = (0, _Auth.useAuth)();
const { setStepNav } = (0, _StepNav.useStepNav)();
const { getPreference, setPreference } = (0, _Preferences.usePreferences)();
const { limit, page, search, sort, where } = (0, _SearchParams.useSearchParams)();
const history = (0, _reactrouterdom.useHistory)();
const { t } = (0, _reacti18next.useTranslation)('general');
const [fetchURL, setFetchURL] = (0, _react.useState)('');
const [fields] = (0, _react.useState)(()=>(0, _formatFields.default)(collection));
const collectionPermissions = permissions?.collections?.[slug];
const hasCreatePermission = collectionPermissions?.create?.permission;
const newDocumentURL = `${admin}/collections/${slug}/create`;
const [{ data }, { setParams }] = (0, _usePayloadAPI.default)(fetchURL, {
initialParams: {
page: 1
}
});
const titleField = (0, _useUseAsTitle.useUseTitleField)(collection);
(0, _react.useEffect)(()=>{
if (CustomList && typeof CustomList === 'object' && 'actions' in CustomList) {
setViewActions(CustomList.actions || []);
}
return ()=>{
setViewActions([]);
};
}, [
CustomList,
setViewActions
]);
(0, _react.useEffect)(()=>{
setStepNav([
{
label: plural
}
]);
}, [
setStepNav,
plural
]);
// /////////////////////////////////////
// Set up Payload REST API query params
// /////////////////////////////////////
const resetParams = (0, _react.useCallback)((overrides = {})=>{
const params = {
depth: 0,
draft: 'true',
limit,
page: overrides?.page,
search: overrides?.search,
sort: overrides?.sort,
where: overrides?.where || {}
};
if (page) params.page = page;
if (sort) params.sort = sort;
if (where) params.where = where;
params.invoke = (0, _uuid.v4)();
if (search) {
let copyOfWhere = {
...where || {}
};
const searchAsConditions = (listSearchableFields || [
titleField?.name || 'id'
]).map((fieldName)=>{
return {
[fieldName]: {
like: search
}
};
}, []);
if (searchAsConditions.length > 0) {
const conditionalSearchFields = {
or: [
...searchAsConditions
]
};
copyOfWhere = hoistQueryParamsToAnd(copyOfWhere, conditionalSearchFields);
}
params.where = copyOfWhere;
}
setParams(params);
}, [
limit,
page,
setParams,
sort,
where,
search,
listSearchableFields,
titleField?.name
]);
(0, _react.useEffect)(()=>{
// Performance enhancement
// Setting the Fetch URL this way
// prevents a double-fetch
setFetchURL(`${serverURL}${api}/${slug}`);
resetParams();
}, [
api,
resetParams,
serverURL,
slug
]);
// /////////////////////////////////////
// Fetch preferences on first load
// /////////////////////////////////////
(0, _react.useEffect)(()=>{
(async ()=>{
const currentPreferences = await getPreference(preferenceKey);
const params = _qs.default.parse(history.location.search, {
depth: 0,
ignoreQueryPrefix: true
});
const search = {
...params,
limit: params?.limit || currentPreferences?.limit || defaultLimit,
sort: params?.sort || currentPreferences?.sort
};
const newSearchQuery = _qs.default.stringify(search, {
addQueryPrefix: true
});
if (newSearchQuery.length > 1) {
history.replace({
search: newSearchQuery
});
}
})();
}, [
collection,
getPreference,
preferenceKey,
history,
t,
defaultLimit
]);
// /////////////////////////////////////
// Set preferences on change
// /////////////////////////////////////
(0, _react.useEffect)(()=>{
void setPreference(preferenceKey, {
sort
}, true);
}, [
sort,
preferenceKey,
setPreference
]);
(0, _react.useEffect)(()=>{
void setPreference(preferenceKey, {
limit
}, true);
}, [
limit,
preferenceKey,
setPreference
]);
// /////////////////////////////////////
// Prevent going beyond page limit
// /////////////////////////////////////
(0, _react.useEffect)(()=>{
if (data?.totalDocs && data.pagingCounter > data.totalDocs) {
const params = _qs.default.parse(history.location.search, {
depth: 0,
ignoreQueryPrefix: true
});
const newSearchQuery = _qs.default.stringify({
...params,
page: data.totalPages
}, {
addQueryPrefix: true
});
history.replace({
search: newSearchQuery
});
}
}, [
data,
history,
resetParams
]);
let ListToRender = null;
if (CustomList && typeof CustomList === 'function') {
ListToRender = CustomList;
} else if (typeof CustomList === 'object' && typeof CustomList.Component === 'function') {
ListToRender = CustomList.Component;
}
return /*#__PURE__*/ _react.default.createElement(_TableColumns.TableColumnsProvider, {
collection: collection
}, /*#__PURE__*/ _react.default.createElement(_RenderCustomComponent.default, {
CustomComponent: ListToRender,
DefaultComponent: _Default.default,
componentProps: {
collection: {
...collection,
fields
},
data,
hasCreatePermission,
limit: limit || defaultLimit,
newDocumentURL,
resetParams,
titleField
}
}));
};
const _default = ListView;
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../../../../src/admin/components/views/collections/List/index.tsx"],"sourcesContent":["import queryString from 'qs'\nimport React, { useCallback, useEffect, useState } from 'react'\nimport { useTranslation } from 'react-i18next'\nimport { useHistory } from 'react-router-dom'\nimport { v4 as uuid } from 'uuid'\n\nimport type { Where } from '../../../../../exports/types'\nimport type { ListIndexProps, ListPreferences, Props } from './types'\n\nimport { type Field } from '../../../../../fields/config/types'\nimport usePayloadAPI from '../../../../hooks/usePayloadAPI'\nimport { useUseTitleField } from '../../../../hooks/useUseAsTitle'\nimport { useStepNav } from '../../../elements/StepNav'\nimport { TableColumnsProvider } from '../../../elements/TableColumns'\nimport { useActions } from '../../../utilities/ActionsProvider'\nimport { useAuth } from '../../../utilities/Auth'\nimport { useConfig } from '../../../utilities/Config'\nimport { usePreferences } from '../../../utilities/Preferences'\nimport RenderCustomComponent from '../../../utilities/RenderCustomComponent'\nimport { useSearchParams } from '../../../utilities/SearchParams'\nimport DefaultList from './Default'\nimport formatFields from './formatFields'\n\nconst hoistQueryParamsToAnd = (where: Where, queryParams: Where) => {\n  if ('and' in where) {\n    where.and.push(queryParams)\n  } else if ('or' in where) {\n    where = {\n      and: [where, queryParams],\n    }\n  } else {\n    where = {\n      and: [where, queryParams],\n    }\n  }\n\n  return where\n}\n\n/**\n * The ListView component is table which lists the collection's documents.\n * The default list view can be found at the {@link DefaultList} component.\n * Users can also create pass their own custom list view component instead\n * of using the default one.\n */\nconst ListView: React.FC<ListIndexProps> = (props) => {\n  const {\n    collection,\n    collection: {\n      admin: {\n        components: { views: { List: CustomList } = {} } = {},\n        listSearchableFields,\n        pagination: { defaultLimit },\n      },\n      labels: { plural },\n      slug,\n    },\n  } = props\n\n  const {\n    routes: { admin, api },\n    serverURL,\n  } = useConfig()\n\n  const { setViewActions } = useActions()\n\n  const preferenceKey = `${collection.slug}-list`\n  const { permissions } = useAuth()\n  const { setStepNav } = useStepNav()\n  const { getPreference, setPreference } = usePreferences()\n  const { limit, page, search, sort, where } = useSearchParams()\n  const history = useHistory()\n  const { t } = useTranslation('general')\n  const [fetchURL, setFetchURL] = useState<string>('')\n  const [fields] = useState<Field[]>(() => formatFields(collection))\n  const collectionPermissions = permissions?.collections?.[slug]\n  const hasCreatePermission = collectionPermissions?.create?.permission\n  const newDocumentURL = `${admin}/collections/${slug}/create`\n  const [{ data }, { setParams }] = usePayloadAPI(fetchURL, { initialParams: { page: 1 } })\n  const titleField = useUseTitleField(collection)\n\n  useEffect(() => {\n    if (CustomList && typeof CustomList === 'object' && 'actions' in CustomList) {\n      setViewActions(CustomList.actions || [])\n    }\n\n    return () => {\n      setViewActions([])\n    }\n  }, [CustomList, setViewActions])\n\n  useEffect(() => {\n    setStepNav([\n      {\n        label: plural,\n      },\n    ])\n  }, [setStepNav, plural])\n\n  // /////////////////////////////////////\n  // Set up Payload REST API query params\n  // /////////////////////////////////////\n\n  const resetParams = useCallback<Props['resetParams']>(\n    (overrides = {}) => {\n      const params: Record<string, unknown> & { where?: Where } = {\n        depth: 0,\n        draft: 'true',\n        limit,\n        page: overrides?.page,\n        search: overrides?.search,\n        sort: overrides?.sort,\n        where: overrides?.where || {},\n      }\n\n      if (page) params.page = page\n      if (sort) params.sort = sort\n      if (where) params.where = where as Where\n      params.invoke = uuid()\n\n      if (search) {\n        let copyOfWhere = { ...((where as Where) || {}) }\n\n        const searchAsConditions = (listSearchableFields || [titleField?.name || 'id']).map(\n          (fieldName) => {\n            return {\n              [fieldName]: {\n                like: search,\n              },\n            }\n          },\n          [],\n        )\n\n        if (searchAsConditions.length > 0) {\n          const conditionalSearchFields = {\n            or: [...searchAsConditions],\n          }\n          copyOfWhere = hoistQueryParamsToAnd(copyOfWhere, conditionalSearchFields)\n        }\n\n        params.where = copyOfWhere\n      }\n\n      setParams(params)\n    },\n    [limit, page, setParams, sort, where, search, listSearchableFields, titleField?.name],\n  )\n\n  useEffect(() => {\n    // Performance enhancement\n    // Setting the Fetch URL this way\n    // prevents a double-fetch\n    setFetchURL(`${serverURL}${api}/${slug}`)\n    resetParams()\n  }, [api, resetParams, serverURL, slug])\n\n  // /////////////////////////////////////\n  // Fetch preferences on first load\n  // /////////////////////////////////////\n\n  useEffect(() => {\n    ;(async () => {\n      const currentPreferences = await getPreference<ListPreferences>(preferenceKey)\n\n      const params = queryString.parse(history.location.search, {\n        depth: 0,\n        ignoreQueryPrefix: true,\n      })\n\n      const search = {\n        ...params,\n        limit: params?.limit || currentPreferences?.limit || defaultLimit,\n        sort: params?.sort || currentPreferences?.sort,\n      }\n\n      const newSearchQuery = queryString.stringify(search, { addQueryPrefix: true })\n\n      if (newSearchQuery.length > 1) {\n        history.replace({\n          search: newSearchQuery,\n        })\n      }\n    })()\n  }, [collection, getPreference, preferenceKey, history, t, defaultLimit])\n\n  // /////////////////////////////////////\n  // Set preferences on change\n  // /////////////////////////////////////\n\n  useEffect(() => {\n    void setPreference(preferenceKey, { sort }, true)\n  }, [sort, preferenceKey, setPreference])\n\n  useEffect(() => {\n    void setPreference(preferenceKey, { limit }, true)\n  }, [limit, preferenceKey, setPreference])\n\n  // /////////////////////////////////////\n  // Prevent going beyond page limit\n  // /////////////////////////////////////\n\n  useEffect(() => {\n    if (data?.totalDocs && data.pagingCounter > data.totalDocs) {\n      const params = queryString.parse(history.location.search, {\n        depth: 0,\n        ignoreQueryPrefix: true,\n      })\n      const newSearchQuery = queryString.stringify(\n        {\n          ...params,\n          page: data.totalPages,\n        },\n        { addQueryPrefix: true },\n      )\n      history.replace({\n        search: newSearchQuery,\n      })\n    }\n  }, [data, history, resetParams])\n\n  let ListToRender = null\n\n  if (CustomList && typeof CustomList === 'function') {\n    ListToRender = CustomList\n  } else if (typeof CustomList === 'object' && typeof CustomList.Component === 'function') {\n    ListToRender = CustomList.Component\n  }\n\n  return (\n    <TableColumnsProvider collection={collection}>\n      <RenderCustomComponent\n        CustomComponent={ListToRender}\n        DefaultComponent={DefaultList}\n        componentProps={{\n          collection: { ...collection, fields },\n          data,\n          hasCreatePermission,\n          limit: limit || defaultLimit,\n          newDocumentURL,\n          resetParams,\n          titleField,\n        }}\n      />\n    </TableColumnsProvider>\n  )\n}\n\nexport default ListView\n"],"names":["hoistQueryParamsToAnd","where","queryParams","and","push","ListView","props","collection","admin","components","views","List","CustomList","listSearchableFields","pagination","defaultLimit","labels","plural","slug","routes","api","serverURL","useConfig","setViewActions","useActions","preferenceKey","permissions","useAuth","setStepNav","useStepNav","getPreference","setPreference","usePreferences","limit","page","search","sort","useSearchParams","history","useHistory","t","useTranslation","fetchURL","setFetchURL","useState","fields","formatFields","collectionPermissions","collections","hasCreatePermission","create","permission","newDocumentURL","data","setParams","usePayloadAPI","initialParams","titleField","useUseTitleField","useEffect","actions","label","resetParams","useCallback","overrides","params","depth","draft","invoke","uuid","copyOfWhere","searchAsConditions","name","map","fieldName","like","length","conditionalSearchFields","or","currentPreferences","queryString","parse","location","ignoreQueryPrefix","newSearchQuery","stringify","addQueryPrefix","replace","totalDocs","pagingCounter","totalPages","ListToRender","Component","TableColumnsProvider","RenderCustomComponent","CustomComponent","DefaultComponent","DefaultList","componentProps"],"mappings":"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"}