payload
Version:
Node, React and MongoDB Headless CMS and Application Framework
183 lines (182 loc) • 17.9 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", {
value: true
});
function _export(target, all) {
for(var name in all)Object.defineProperty(target, name, {
enumerable: true,
get: all[name]
});
}
_export(exports, {
PreferencesProvider: function() {
return PreferencesProvider;
},
usePreferences: function() {
return usePreferences;
}
});
const _deepequal = /*#__PURE__*/ _interop_require_default(require("deep-equal"));
const _react = /*#__PURE__*/ _interop_require_wildcard(require("react"));
const _reacti18next = require("react-i18next");
const _api = require("../../../api");
const _Auth = require("../Auth");
const _Config = require("../Config");
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 Context = /*#__PURE__*/ (0, _react.createContext)({});
const requestOptions = (value, language)=>({
body: JSON.stringify({
value
}),
headers: {
'Accept-Language': language,
'Content-Type': 'application/json'
}
});
const PreferencesProvider = ({ children })=>{
const contextRef = (0, _react.useRef)({});
const preferencesRef = (0, _react.useRef)({});
const pendingUpdate = (0, _react.useRef)({});
const config = (0, _Config.useConfig)();
const { user } = (0, _Auth.useAuth)();
const { i18n } = (0, _reacti18next.useTranslation)();
const { routes: { api }, serverURL } = config;
(0, _react.useEffect)(()=>{
if (!user) {
// clear preferences between users
preferencesRef.current = {};
}
}, [
user
]);
const getPreference = (0, _react.useCallback)(async (key)=>{
const prefs = preferencesRef.current;
if (typeof prefs[key] !== 'undefined') return prefs[key];
const promise = new Promise((resolve)=>{
void (async ()=>{
const request = await _api.requests.get(`${serverURL}${api}/payload-preferences/${key}`, {
headers: {
'Accept-Language': i18n.language
}
});
let value = null;
if (request.status === 200) {
const preference = await request.json();
value = preference.value;
}
preferencesRef.current[key] = value;
resolve(value);
})();
});
prefs[key] = promise;
return promise;
}, [
i18n.language,
api,
preferencesRef,
serverURL
]);
const setPreference = (0, _react.useCallback)(async (key, value, merge = false)=>{
if (merge === false) {
preferencesRef.current[key] = value;
await _api.requests.post(`${serverURL}${api}/payload-preferences/${key}`, requestOptions(value, i18n.language));
return;
}
let newValue = value;
const currentPreference = await getPreference(key);
// handle value objects where multiple values can be set under one key
if (typeof value === 'object' && typeof currentPreference === 'object' && typeof newValue === 'object') {
// merge the value with any existing preference for the key
newValue = {
...currentPreference || {},
...value
};
if ((0, _deepequal.default)(newValue, currentPreference)) {
return;
}
// add the requested changes to a pendingUpdate batch for the key
pendingUpdate.current[key] = {
...pendingUpdate.current[key],
...newValue
};
} else {
if (newValue === currentPreference) {
return;
}
pendingUpdate.current[key] = newValue;
}
const updatePreference = async ()=>{
// compare the value stored in context before sending to eliminate duplicate requests
if ((0, _deepequal.default)(pendingUpdate.current[key], preferencesRef.current[key])) {
return;
}
// preference set in context here to prevent other updatePreference at the same time
preferencesRef.current[key] = pendingUpdate.current[key];
await _api.requests.post(`${serverURL}${api}/payload-preferences/${key}`, requestOptions(preferencesRef.current[key], i18n.language));
// reset any changes for this key after sending the request
delete pendingUpdate.current[key];
};
// use timeout to allow multiple changes of different values using the same key in one request
setTimeout(()=>{
void updatePreference();
});
}, [
api,
getPreference,
i18n.language,
pendingUpdate,
serverURL
]);
contextRef.current.getPreference = getPreference;
contextRef.current.setPreference = setPreference;
return /*#__PURE__*/ _react.default.createElement(Context.Provider, {
value: contextRef.current
}, children);
};
const usePreferences = ()=>(0, _react.useContext)(Context);
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../../../src/admin/components/utilities/Preferences/index.tsx"],"sourcesContent":["import isDeepEqual from 'deep-equal'\nimport React, { createContext, useCallback, useContext, useEffect, useRef } from 'react'\nimport { useTranslation } from 'react-i18next'\n\nimport { requests } from '../../../api'\nimport { useAuth } from '../Auth'\nimport { useConfig } from '../Config'\n\ntype PreferencesContext = {\n  getPreference: <T = any>(key: string) => Promise<T> | T\n  /**\n   * @param key - a string identifier for the property being set\n   * @param value - preference data to store\n   * @param merge - when true will combine the existing preference object batch the change into one request for objects, default = false\n   */\n  setPreference: <T = any>(key: string, value: T, merge?: boolean) => Promise<void>\n}\n\nconst Context = createContext({} as PreferencesContext)\n\nconst requestOptions = (value, language) => ({\n  body: JSON.stringify({ value }),\n  headers: {\n    'Accept-Language': language,\n    'Content-Type': 'application/json',\n  },\n})\n\nexport const PreferencesProvider: React.FC<{ children?: React.ReactNode }> = ({ children }) => {\n  const contextRef = useRef({} as PreferencesContext)\n  const preferencesRef = useRef({})\n  const pendingUpdate = useRef({})\n  const config = useConfig()\n  const { user } = useAuth()\n  const { i18n } = useTranslation()\n  const {\n    routes: { api },\n    serverURL,\n  } = config\n\n  useEffect(() => {\n    if (!user) {\n      // clear preferences between users\n      preferencesRef.current = {}\n    }\n  }, [user])\n\n  const getPreference = useCallback(\n    async <T = any,>(key: string): Promise<T> => {\n      const prefs = preferencesRef.current\n      if (typeof prefs[key] !== 'undefined') return prefs[key]\n      const promise = new Promise((resolve: (value: T) => void) => {\n        void (async () => {\n          const request = await requests.get(`${serverURL}${api}/payload-preferences/${key}`, {\n            headers: {\n              'Accept-Language': i18n.language,\n            },\n          })\n          let value = null\n          if (request.status === 200) {\n            const preference = await request.json()\n            value = preference.value\n          }\n          preferencesRef.current[key] = value\n          resolve(value)\n        })()\n      })\n      prefs[key] = promise\n      return promise\n    },\n    [i18n.language, api, preferencesRef, serverURL],\n  )\n\n  const setPreference = useCallback(\n    async (key: string, value: unknown, merge = false): Promise<void> => {\n      if (merge === false) {\n        preferencesRef.current[key] = value\n        await requests.post(\n          `${serverURL}${api}/payload-preferences/${key}`,\n          requestOptions(value, i18n.language),\n        )\n        return\n      }\n\n      let newValue = value\n      const currentPreference = await getPreference(key)\n      // handle value objects where multiple values can be set under one key\n      if (\n        typeof value === 'object' &&\n        typeof currentPreference === 'object' &&\n        typeof newValue === 'object'\n      ) {\n        // merge the value with any existing preference for the key\n        newValue = { ...(currentPreference || {}), ...value }\n        if (isDeepEqual(newValue, currentPreference)) {\n          return\n        }\n        // add the requested changes to a pendingUpdate batch for the key\n        pendingUpdate.current[key] = {\n          ...pendingUpdate.current[key],\n          ...(newValue as Record<string, unknown>),\n        }\n      } else {\n        if (newValue === currentPreference) {\n          return\n        }\n        pendingUpdate.current[key] = newValue\n      }\n\n      const updatePreference = async () => {\n        // compare the value stored in context before sending to eliminate duplicate requests\n        if (isDeepEqual(pendingUpdate.current[key], preferencesRef.current[key])) {\n          return\n        }\n        // preference set in context here to prevent other updatePreference at the same time\n        preferencesRef.current[key] = pendingUpdate.current[key]\n\n        await requests.post(\n          `${serverURL}${api}/payload-preferences/${key}`,\n          requestOptions(preferencesRef.current[key], i18n.language),\n        )\n        // reset any changes for this key after sending the request\n        delete pendingUpdate.current[key]\n      }\n\n      // use timeout to allow multiple changes of different values using the same key in one request\n      setTimeout(() => {\n        void updatePreference()\n      })\n    },\n    [api, getPreference, i18n.language, pendingUpdate, serverURL],\n  )\n\n  contextRef.current.getPreference = getPreference\n  contextRef.current.setPreference = setPreference\n\n  return <Context.Provider value={contextRef.current}>{children}</Context.Provider>\n}\n\nexport const usePreferences = (): PreferencesContext => useContext(Context)\n"],"names":["PreferencesProvider","usePreferences","Context","createContext","requestOptions","value","language","body","JSON","stringify","headers","children","contextRef","useRef","preferencesRef","pendingUpdate","config","useConfig","user","useAuth","i18n","useTranslation","routes","api","serverURL","useEffect","current","getPreference","useCallback","key","prefs","promise","Promise","resolve","request","requests","get","status","preference","json","setPreference","merge","post","newValue","currentPreference","isDeepEqual","updatePreference","setTimeout","Provider","useContext"],"mappings":"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"}