Files
dbgate/packages/web/src/utility/useEditorData.js
2020-12-26 10:28:49 +01:00

120 lines
3.6 KiB
JavaScript

import React from 'react';
import _ from 'lodash';
import localforage from 'localforage';
import { changeTab } from './common';
import { useSetOpenedTabs } from './globalState';
function getParsedLocalStorage(key) {
const value = localStorage.getItem(key);
if (value != null) {
try {
const res = JSON.parse(value);
return res;
} catch (e) {
localStorage.removeItem(key);
}
}
return null;
}
export default function useEditorData({ tabid, reloadToken = 0, loadFromArgs = null }) {
const localStorageKey = `tabdata_editor_${tabid}`;
const setOpenedTabs = useSetOpenedTabs();
const changeCounterRef = React.useRef(0);
const savedCounterRef = React.useRef(0);
const [errorMessage, setErrorMessage] = React.useState(null);
const [value, setValue] = React.useState(null);
const [isLoading, setIsLoading] = React.useState(true);
const initialDataRef = React.useRef(null);
const valueRef = React.useRef(null);
const initialLoad = async () => {
if (loadFromArgs) {
try {
const init = await loadFromArgs();
changeTab(tabid, setOpenedTabs, (tab) => ({
...tab,
props: _.omit(tab.props, ['initialArgs']),
}));
setValue(init);
valueRef.current = init;
initialDataRef.current = init;
} catch (err) {
const message = (err && err.response && err.response.data && err.response.data.error) || 'Loading failed';
setErrorMessage(message);
console.error(err.response);
}
} else {
const initFallback = getParsedLocalStorage(localStorageKey);
if (initFallback != null) {
const init = JSON.parse(initFallback);
setValue(init);
valueRef.current = init;
// move to local forage
await localforage.setItem(localStorageKey, init);
localStorage.removeItem(localStorageKey);
initialDataRef.current = init;
} else {
const init = await localforage.getItem(localStorageKey);
if (init) {
setValue(init);
valueRef.current = init;
initialDataRef.current = init;
}
}
}
setIsLoading(false);
};
React.useEffect(() => {
initialLoad();
}, [reloadToken]);
const saveToStorage = React.useCallback(async () => {
if (valueRef.current == null) return;
try {
await localforage.setItem(localStorageKey, valueRef.current);
localStorage.removeItem(localStorageKey);
savedCounterRef.current = changeCounterRef.current;
} catch (err) {
console.error(err);
}
}, [localStorageKey, valueRef]);
const saveToStorageSync = React.useCallback(() => {
if (valueRef.current == null) return;
if (savedCounterRef.current == changeCounterRef.current) return; // all saved
// on window unload must be synchronous actions, save to local storage instead
localStorage.setItem(localStorageKey, JSON.stringify(valueRef.current));
}, [localStorageKey, valueRef]);
const saveToStorageDebounced = React.useMemo(() => _.debounce(saveToStorage, 5000), [saveToStorage]);
const handleChange = (newValue) => {
if (newValue != null) valueRef.current = newValue;
setValue(newValue);
changeCounterRef.current += 1;
saveToStorageDebounced();
};
React.useEffect(() => {
window.addEventListener('beforeunload', saveToStorageSync);
return () => {
saveToStorage();
window.removeEventListener('beforeunload', saveToStorageSync);
};
}, []);
return {
editorData: value,
setEditorData: handleChange,
isLoading,
initialData: initialDataRef.current,
errorMessage,
saveToStorage,
saveToStorageSync,
};
}