UNPKG

payload

Version:

Node, React and MongoDB Headless CMS and Application Framework

183 lines (182 loc) • 17.9 kB
"use strict"; 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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"}