payload
Version:
Node, React and MongoDB Headless CMS and Application Framework
209 lines (208 loc) • 20.8 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 _reactrouterdom = require("react-router-dom");
const _reacttoastify = require("react-toastify");
const _useDebounce = /*#__PURE__*/ _interop_require_default(require("../../../hooks/useDebounce"));
const _formatDate = require("../../../utilities/formatDate");
const _context = require("../../forms/Form/context");
const _reduceFieldsToValues = /*#__PURE__*/ _interop_require_default(require("../../forms/Form/reduceFieldsToValues"));
const _Config = require("../../utilities/Config");
const _DocumentInfo = require("../../utilities/DocumentInfo");
const _Locale = require("../../utilities/Locale");
require("./index.scss");
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 = 'autosave';
const Autosave = ({ id, collection, global, publishedDocUpdatedAt })=>{
const { routes: { admin, api }, serverURL } = (0, _Config.useConfig)();
const { getVersions, versions } = (0, _DocumentInfo.useDocumentInfo)();
const [fields] = (0, _context.useAllFormFields)();
const modified = (0, _context.useFormModified)();
const { code: locale } = (0, _Locale.useLocale)();
const { replace } = (0, _reactrouterdom.useHistory)();
const { i18n, t } = (0, _reacti18next.useTranslation)('version');
let interval = 800;
if (collection?.versions.drafts && collection.versions?.drafts?.autosave) interval = collection.versions.drafts.autosave.interval;
if (global?.versions.drafts && global.versions?.drafts?.autosave) interval = global.versions.drafts.autosave.interval;
const [saving, setSaving] = (0, _react.useState)(false);
const [lastSaved, setLastSaved] = (0, _react.useState)();
const debouncedFields = (0, _useDebounce.default)(fields, interval);
const fieldRef = (0, _react.useRef)(fields);
const modifiedRef = (0, _react.useRef)(modified);
const localeRef = (0, _react.useRef)(locale);
// Store fields in ref so the autosave func
// can always retrieve the most to date copies
// after the timeout has executed
fieldRef.current = fields;
// Store modified in ref so the autosave func
// can bail out if modified becomes false while
// timing out during autosave
modifiedRef.current = modified;
const createCollectionDoc = (0, _react.useCallback)(async ()=>{
const res = await fetch(`${serverURL}${api}/${collection.slug}?locale=${locale}&fallback-locale=null&depth=0&draft=true&autosave=true`, {
body: JSON.stringify({}),
credentials: 'include',
headers: {
'Accept-Language': i18n.language,
'Content-Type': 'application/json'
},
method: 'POST'
});
if (res.status === 201) {
const json = await res.json();
replace(`${admin}/collections/${collection.slug}/${json.doc.id}`, {
state: {
data: json.doc
}
});
} else {
_reacttoastify.toast.error(t('error:autosaving'));
}
}, [
i18n,
serverURL,
api,
collection,
locale,
replace,
admin,
t
]);
(0, _react.useEffect)(()=>{
// If no ID, but this is used for a collection doc,
// Immediately save it and set lastSaved
if (!id && collection) {
createCollectionDoc();
}
}, [
id,
collection,
createCollectionDoc
]);
// When debounced fields change, autosave
(0, _react.useEffect)(()=>{
const autosave = async ()=>{
if (modified) {
setSaving(true);
let url;
let method;
if (collection && id) {
url = `${serverURL}${api}/${collection.slug}/${id}?draft=true&autosave=true&locale=${localeRef.current}`;
method = 'PATCH';
}
if (global) {
url = `${serverURL}${api}/globals/${global.slug}?draft=true&autosave=true&locale=${localeRef.current}`;
method = 'POST';
}
if (url) {
setTimeout(async ()=>{
if (modifiedRef.current) {
const body = {
...(0, _reduceFieldsToValues.default)(fieldRef.current, true),
_status: 'draft'
};
const res = await fetch(url, {
body: JSON.stringify(body),
credentials: 'include',
headers: {
'Accept-Language': i18n.language,
'Content-Type': 'application/json'
},
method
});
if (res.status === 200) {
setLastSaved(new Date().getTime());
getVersions();
}
}
setSaving(false);
}, 1000);
}
}
};
autosave();
}, [
i18n,
debouncedFields,
modified,
serverURL,
api,
collection,
global,
id,
getVersions,
localeRef,
modifiedRef
]);
(0, _react.useEffect)(()=>{
if (versions?.docs?.[0]) {
setLastSaved(new Date(versions.docs[0].updatedAt).getTime());
} else if (publishedDocUpdatedAt) {
setLastSaved(new Date(publishedDocUpdatedAt).getTime());
}
}, [
publishedDocUpdatedAt,
versions
]);
return /*#__PURE__*/ _react.default.createElement("div", {
className: baseClass
}, saving && t('saving'), !saving && lastSaved && /*#__PURE__*/ _react.default.createElement(_react.default.Fragment, null, t('lastSavedAgo', {
distance: (0, _formatDate.formatTimeToNow)(lastSaved, i18n.language)
})));
};
const _default = Autosave;
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../../../src/admin/components/elements/Autosave/index.tsx"],"sourcesContent":["import React, { useCallback, useEffect, useRef, useState } from 'react'\nimport { useTranslation } from 'react-i18next'\nimport { useHistory } from 'react-router-dom'\nimport { toast } from 'react-toastify'\n\nimport type { Props } from './types'\n\nimport useDebounce from '../../../hooks/useDebounce'\nimport { formatTimeToNow } from '../../../utilities/formatDate'\nimport { useAllFormFields, useFormModified } from '../../forms/Form/context'\nimport reduceFieldsToValues from '../../forms/Form/reduceFieldsToValues'\nimport { useConfig } from '../../utilities/Config'\nimport { useDocumentInfo } from '../../utilities/DocumentInfo'\nimport { useLocale } from '../../utilities/Locale'\nimport './index.scss'\nconst baseClass = 'autosave'\n\nconst Autosave: React.FC<Props> = ({ id, collection, global, publishedDocUpdatedAt }) => {\n  const {\n    routes: { admin, api },\n    serverURL,\n  } = useConfig()\n  const { getVersions, versions } = useDocumentInfo()\n  const [fields] = useAllFormFields()\n  const modified = useFormModified()\n  const { code: locale } = useLocale()\n  const { replace } = useHistory()\n  const { i18n, t } = useTranslation('version')\n\n  let interval = 800\n  if (collection?.versions.drafts && collection.versions?.drafts?.autosave)\n    interval = collection.versions.drafts.autosave.interval\n  if (global?.versions.drafts && global.versions?.drafts?.autosave)\n    interval = global.versions.drafts.autosave.interval\n\n  const [saving, setSaving] = useState(false)\n  const [lastSaved, setLastSaved] = useState<number>()\n  const debouncedFields = useDebounce(fields, interval)\n  const fieldRef = useRef(fields)\n  const modifiedRef = useRef(modified)\n  const localeRef = useRef(locale)\n\n  // Store fields in ref so the autosave func\n  // can always retrieve the most to date copies\n  // after the timeout has executed\n  fieldRef.current = fields\n\n  // Store modified in ref so the autosave func\n  // can bail out if modified becomes false while\n  // timing out during autosave\n  modifiedRef.current = modified\n\n  const createCollectionDoc = useCallback(async () => {\n    const res = await fetch(\n      `${serverURL}${api}/${collection.slug}?locale=${locale}&fallback-locale=null&depth=0&draft=true&autosave=true`,\n      {\n        body: JSON.stringify({}),\n        credentials: 'include',\n        headers: {\n          'Accept-Language': i18n.language,\n          'Content-Type': 'application/json',\n        },\n        method: 'POST',\n      },\n    )\n\n    if (res.status === 201) {\n      const json = await res.json()\n      replace(`${admin}/collections/${collection.slug}/${json.doc.id}`, {\n        state: {\n          data: json.doc,\n        },\n      })\n    } else {\n      toast.error(t('error:autosaving'))\n    }\n  }, [i18n, serverURL, api, collection, locale, replace, admin, t])\n\n  useEffect(() => {\n    // If no ID, but this is used for a collection doc,\n    // Immediately save it and set lastSaved\n    if (!id && collection) {\n      createCollectionDoc()\n    }\n  }, [id, collection, createCollectionDoc])\n\n  // When debounced fields change, autosave\n\n  useEffect(() => {\n    const autosave = async () => {\n      if (modified) {\n        setSaving(true)\n\n        let url: string\n        let method: string\n\n        if (collection && id) {\n          url = `${serverURL}${api}/${collection.slug}/${id}?draft=true&autosave=true&locale=${localeRef.current}`\n          method = 'PATCH'\n        }\n\n        if (global) {\n          url = `${serverURL}${api}/globals/${global.slug}?draft=true&autosave=true&locale=${localeRef.current}`\n          method = 'POST'\n        }\n\n        if (url) {\n          setTimeout(async () => {\n            if (modifiedRef.current) {\n              const body = {\n                ...reduceFieldsToValues(fieldRef.current, true),\n                _status: 'draft',\n              }\n\n              const res = await fetch(url, {\n                body: JSON.stringify(body),\n                credentials: 'include',\n                headers: {\n                  'Accept-Language': i18n.language,\n                  'Content-Type': 'application/json',\n                },\n                method,\n              })\n\n              if (res.status === 200) {\n                setLastSaved(new Date().getTime())\n                getVersions()\n              }\n            }\n\n            setSaving(false)\n          }, 1000)\n        }\n      }\n    }\n\n    autosave()\n  }, [\n    i18n,\n    debouncedFields,\n    modified,\n    serverURL,\n    api,\n    collection,\n    global,\n    id,\n    getVersions,\n    localeRef,\n    modifiedRef,\n  ])\n\n  useEffect(() => {\n    if (versions?.docs?.[0]) {\n      setLastSaved(new Date(versions.docs[0].updatedAt).getTime())\n    } else if (publishedDocUpdatedAt) {\n      setLastSaved(new Date(publishedDocUpdatedAt).getTime())\n    }\n  }, [publishedDocUpdatedAt, versions])\n\n  return (\n    <div className={baseClass}>\n      {saving && t('saving')}\n      {!saving && lastSaved && (\n        <React.Fragment>\n          {t('lastSavedAgo', {\n            distance: formatTimeToNow(lastSaved, i18n.language),\n          })}\n        </React.Fragment>\n      )}\n    </div>\n  )\n}\n\nexport default Autosave\n"],"names":["baseClass","Autosave","id","collection","global","publishedDocUpdatedAt","routes","admin","api","serverURL","useConfig","getVersions","versions","useDocumentInfo","fields","useAllFormFields","modified","useFormModified","code","locale","useLocale","replace","useHistory","i18n","t","useTranslation","interval","drafts","autosave","saving","setSaving","useState","lastSaved","setLastSaved","debouncedFields","useDebounce","fieldRef","useRef","modifiedRef","localeRef","current","createCollectionDoc","useCallback","res","fetch","slug","body","JSON","stringify","credentials","headers","language","method","status","json","doc","state","data","toast","error","useEffect","url","setTimeout","reduceFieldsToValues","_status","Date","getTime","docs","updatedAt","div","className","React","Fragment","distance","formatTimeToNow"],"mappings":"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"}