react-live-chat-loader
Version:
Implement live chat in your react app without taking a performance hit.
98 lines (96 loc) • 12.7 kB
JavaScript
import { useContext, useCallback, useEffect } from 'react';
import { LiveChatLoaderContext } from "../context";
import * as Providers from "../providers";
var requestIdleCallback = typeof window !== 'undefined' ? window.requestIdleCallback : null;
var connection = typeof window !== 'undefined' ?
// eslint-disable-next-line @typescript-eslint/no-explicit-any
window.navigator && window.navigator.connection : null;
var useChat = function useChat() {
var _ref = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {
loadWhenIdle: false
},
loadWhenIdle = _ref.loadWhenIdle;
var _useContext = useContext(LiveChatLoaderContext),
provider = _useContext.provider,
providerKey = _useContext.providerKey,
idlePeriod = _useContext.idlePeriod,
state = _useContext.state,
setState = _useContext.setState,
appID = _useContext.appID,
locale = _useContext.locale,
baseUrl = _useContext.baseUrl,
instanceId = _useContext.instanceId,
env = _useContext.env,
geo = _useContext.geo,
beforeInit = _useContext.beforeInit,
onReady = _useContext.onReady;
useEffect(function () {
// Don't load if idlePeriod is 0, null or undefined
if (typeof window === 'undefined' || !loadWhenIdle || !idlePeriod) return;
// Don't load if 2g connection or save-data is enabled
if (connection && (connection.saveData || /2g/.test(connection.effectiveType))) return;
if (isNaN(idlePeriod)) return;
// deadline.timeRemaining() has an upper limit of 50 milliseconds
// We want to ensure the page has been idle for a significant period of time
// Therefore we count consecutive maximum timeRemaining counts and load chat when we reach our threshold
var elapsedIdlePeriod = 0;
var previousTimeRemaining = 0;
var _scheduleLoadChat = function scheduleLoadChat(deadline) {
if (elapsedIdlePeriod > idlePeriod) return loadChat({
open: false
});
var timeRemaining = deadline.timeRemaining();
// To ensure browser is idle, only accumalte elapsedIdlePeriod when
// two consecutive maximum timeRemaining's have been observed
if (previousTimeRemaining > 49 && timeRemaining > 49) elapsedIdlePeriod += timeRemaining;
previousTimeRemaining = timeRemaining;
requestIdleCallback === null || requestIdleCallback === void 0 || requestIdleCallback(_scheduleLoadChat);
};
if (requestIdleCallback) {
requestIdleCallback(_scheduleLoadChat);
} else {
setTimeout(function () {
return loadChat({
open: false
});
}, idlePeriod);
}
}, []);
var chatProvider = Providers[provider];
var loadChat = useCallback(function () {
var _ref2 = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {
open: true
},
_ref2$open = _ref2.open,
open = _ref2$open === void 0 ? true : _ref2$open;
if (!providerKey) {
//eslint-disable-next-line no-console
console.error('No api key given to react-live-chat-loader');
return;
}
if (!provider) {
//eslint-disable-next-line no-console
console.error('No provider given to react-live-chat-loader');
return;
}
chatProvider.load({
providerKey: providerKey,
setState: setState,
appID: appID,
locale: locale,
baseUrl: baseUrl,
instanceId: instanceId,
env: env,
geo: geo,
beforeInit: beforeInit,
onReady: onReady
});
if (open) {
chatProvider.open();
if (state !== 'complete') setState('open');
}
}, [state]);
return [state, loadChat];
};
export default useChat;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["useContext","useCallback","useEffect","LiveChatLoaderContext","Providers","requestIdleCallback","window","connection","navigator","useChat","_ref","arguments","length","undefined","loadWhenIdle","_useContext","provider","providerKey","idlePeriod","state","setState","appID","locale","baseUrl","instanceId","env","geo","beforeInit","onReady","saveData","test","effectiveType","isNaN","elapsedIdlePeriod","previousTimeRemaining","scheduleLoadChat","deadline","loadChat","open","timeRemaining","setTimeout","chatProvider","_ref2","_ref2$open","console","error","load"],"sources":["../../src/hooks/useChat.ts"],"sourcesContent":["import { useContext, useCallback, useEffect } from 'react'\n\nimport { State } from '../types'\nimport { LiveChatLoaderContext } from '../context'\nimport * as Providers from '../providers'\n\nconst requestIdleCallback =\n  typeof window !== 'undefined' ? window.requestIdleCallback : null\nconst connection =\n  typeof window !== 'undefined'\n    ? // eslint-disable-next-line @typescript-eslint/no-explicit-any\n      window.navigator && (window.navigator as any).connection\n    : null\n\nconst useChat = (\n  {\n    loadWhenIdle\n  }: {\n    loadWhenIdle: boolean\n  } = { loadWhenIdle: false }\n): [State, ({ open }: { open: boolean }) => void] => {\n  const {\n    provider,\n    providerKey,\n    idlePeriod,\n    state,\n    setState,\n    appID,\n    locale,\n    baseUrl,\n    // instanceId, env and geo are only relevant for Adobe Dynamic Chat\n    instanceId,\n    env,\n    geo,\n    beforeInit,\n    onReady\n  } = useContext(LiveChatLoaderContext)\n\n  useEffect(() => {\n    // Don't load if idlePeriod is 0, null or undefined\n    if (typeof window === 'undefined' || !loadWhenIdle || !idlePeriod) return\n\n    // Don't load if 2g connection or save-data is enabled\n    if (\n      connection &&\n      (connection.saveData || /2g/.test(connection.effectiveType))\n    )\n      return\n\n    if (isNaN(idlePeriod)) return\n\n    // deadline.timeRemaining() has an upper limit of 50 milliseconds\n    // We want to ensure the page has been idle for a significant period of time\n    // Therefore we count consecutive maximum timeRemaining counts and load chat when we reach our threshold\n    let elapsedIdlePeriod = 0\n    let previousTimeRemaining = 0\n    const scheduleLoadChat = (deadline: IdleDeadline) => {\n      if (elapsedIdlePeriod > idlePeriod) return loadChat({ open: false })\n\n      const timeRemaining = deadline.timeRemaining()\n      // To ensure browser is idle, only accumalte elapsedIdlePeriod when\n      // two consecutive maximum timeRemaining's have been observed\n      if (previousTimeRemaining > 49 && timeRemaining > 49)\n        elapsedIdlePeriod += timeRemaining\n\n      previousTimeRemaining = timeRemaining\n      requestIdleCallback?.(scheduleLoadChat)\n    }\n\n    if (requestIdleCallback) {\n      requestIdleCallback(scheduleLoadChat)\n    } else {\n      setTimeout(() => loadChat({ open: false }), idlePeriod)\n    }\n  }, [])\n\n  const chatProvider = Providers[provider]\n\n  const loadChat = useCallback<(args: { open: boolean }) => void>(\n    ({ open = true } = { open: true }) => {\n      if (!providerKey) {\n        //eslint-disable-next-line no-console\n        console.error('No api key given to react-live-chat-loader')\n        return\n      }\n\n      if (!provider) {\n        //eslint-disable-next-line no-console\n        console.error('No provider given to react-live-chat-loader')\n        return\n      }\n\n      chatProvider.load({\n        providerKey,\n        setState,\n        appID,\n        locale,\n        baseUrl,\n        instanceId,\n        env,\n        geo,\n        beforeInit,\n        onReady\n      })\n\n      if (open) {\n        chatProvider.open()\n        if (state !== 'complete') setState('open')\n      }\n    },\n    [state]\n  )\n\n  return [state, loadChat]\n}\n\nexport default useChat\n"],"mappings":"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","ignoreList":[]}