0
0
mirror of https://github.com/naturalcrit/homebrewery.git synced 2026-01-14 21:32:41 +00:00
This commit is contained in:
Trevor Buckner
2025-09-09 22:35:55 -04:00
parent 1044aa74b0
commit 95a1d74644

View File

@@ -5,12 +5,11 @@ import React, { useState, useEffect, useRef, useCallback } from 'react';
import request from '../../utils/request-middleware.js'; import request from '../../utils/request-middleware.js';
import Markdown from 'naturalcrit/markdown.js'; import Markdown from 'naturalcrit/markdown.js';
import _ from 'lodash';; import _ from 'lodash';;
import {makePatches, stringifyPatches} from '@sanity/diff-match-patch'; import { makePatches, stringifyPatches } from '@sanity/diff-match-patch';
import { md5 } from 'hash-wasm'; import { md5 } from 'hash-wasm';
import { gzipSync, strToU8 } from 'fflate'; import { gzipSync, strToU8 } from 'fflate';
import { Meta } from 'vitreum/headtags';
const { Meta } = require('vitreum/headtags');
import Nav from 'naturalcrit/nav/nav.jsx'; import Nav from 'naturalcrit/nav/nav.jsx';
import Navbar from '../../navbar/navbar.jsx'; import Navbar from '../../navbar/navbar.jsx';
@@ -39,9 +38,9 @@ const SAVE_TIMEOUT = 10000;
const UNSAVED_WARNING_TIMEOUT = 900000; //Warn user afer 15 minutes of unsaved changes const UNSAVED_WARNING_TIMEOUT = 900000; //Warn user afer 15 minutes of unsaved changes
const UNSAVED_WARNING_POPUP_TIMEOUT = 4000; //Show the warning for 4 seconds const UNSAVED_WARNING_POPUP_TIMEOUT = 4000; //Show the warning for 4 seconds
const EditPage = (props) => { const EditPage = (props)=>{
props = { props = {
brew: DEFAULT_BREW_LOAD, brew : DEFAULT_BREW_LOAD,
...props ...props
}; };
@@ -64,141 +63,139 @@ const EditPage = (props) => {
const [unsavedTime , setUnsavedTime ] = useState(new Date()); const [unsavedTime , setUnsavedTime ] = useState(new Date());
const editorRef = useRef(null); const editorRef = useRef(null);
const savedBrew = useRef(_.cloneDeep(props.brew)); const savedBrew = useRef(_.cloneDeep(props.brew));
const warningTimer = useRef(null);
const debounceSave = useCallback(_.debounce((brew, saveToGoogle)=>save(brew, saveToGoogle), SAVE_TIMEOUT), []); const debounceSave = useCallback(_.debounce((brew, saveToGoogle)=>save(brew, saveToGoogle), SAVE_TIMEOUT), []);
useEffect(() => { useEffect(()=>{
setUrl(window.location.href); setUrl(window.location.href);
const autoSavePref = JSON.parse(localStorage.getItem('AUTOSAVE_ON') ?? true); const autoSavePref = JSON.parse(localStorage.getItem('AUTOSAVE_ON') ?? true);
setAutoSaveEnabled(autoSavePref); setAutoSaveEnabled(autoSavePref);
setAutoSaveWarning(!autoSavePref) setAutoSaveWarning(!autoSavePref);
setHTMLErrors(Markdown.validate(currentBrew.text)); setHTMLErrors(Markdown.validate(currentBrew.text));
fetchThemeBundle(setError, setThemeBundle, currentBrew.renderer, currentBrew.theme); fetchThemeBundle(setError, setThemeBundle, currentBrew.renderer, currentBrew.theme);
document.addEventListener('keydown', handleControlKeys); document.addEventListener('keydown', handleControlKeys);
window.onbeforeunload = () => { window.onbeforeunload = ()=>{
if (isSaving || unsavedChanges) if(isSaving || unsavedChanges)
return 'You have unsaved changes!'; return 'You have unsaved changes!';
}; };
return () => { return ()=>{
document.removeEventListener('keydown', handleControlKeys); document.removeEventListener('keydown', handleControlKeys);
window.onbeforeunload = null; window.onbeforeunload = null;
}; };
}, []); }, []);
useEffect(() => { useEffect(()=>{
const hasChange = !_.isEqual(currentBrew, savedBrew.current); const hasChange = !_.isEqual(currentBrew, savedBrew.current);
setUnsavedChanges(hasChange); setUnsavedChanges(hasChange);
if(hasChange && autoSaveEnabled) trySave(); if(hasChange && autoSaveEnabled) trySave();
}, [currentBrew]); }, [currentBrew]);
const handleControlKeys = (e) => { const handleControlKeys = (e)=>{
if (!(e.ctrlKey || e.metaKey)) return; if(!(e.ctrlKey || e.metaKey)) return;
const S_KEY = 83; const S_KEY = 83;
const P_KEY = 80; const P_KEY = 80;
if (e.keyCode === S_KEY) trySave(true); if(e.keyCode === S_KEY) trySave(true);
if (e.keyCode === P_KEY) printCurrentBrew(); if(e.keyCode === P_KEY) printCurrentBrew();
if (e.keyCode === S_KEY || e.keyCode === P_KEY) { if(e.keyCode === S_KEY || e.keyCode === P_KEY) {
e.stopPropagation(); e.stopPropagation();
e.preventDefault(); e.preventDefault();
} }
}; };
const handleSplitMove = () => { const handleSplitMove = ()=>{
editorRef.current?.update(); editorRef.current?.update();
}; };
const handleEditorViewPageChange = (pageNumber) => { const handleEditorViewPageChange = (pageNumber)=>{
setCurrentEditorViewPageNum(pageNumber); setCurrentEditorViewPageNum(pageNumber);
}; };
const handleEditorCursorPageChange = (pageNumber) => { const handleEditorCursorPageChange = (pageNumber)=>{
setCurrentEditorCursorPageNum(pageNumber); setCurrentEditorCursorPageNum(pageNumber);
}; };
const handleBrewRendererPageChange = (pageNumber) => { const handleBrewRendererPageChange = (pageNumber)=>{
setCurrentBrewRendererPageNum(pageNumber); setCurrentBrewRendererPageNum(pageNumber);
}; };
const handleTextChange = (text) => { const handleTextChange = (text)=>{
//If there are HTML errors, run the validator on every change to give quick feedback //If there are HTML errors, run the validator on every change to give quick feedback
if(HTMLErrors.length) if(HTMLErrors.length)
setHTMLErrors(Markdown.validate(text)); setHTMLErrors(Markdown.validate(text));
setCurrentBrew((prevBrew) => ({ ...prevBrew, text })); setCurrentBrew((prevBrew)=>({ ...prevBrew, text }));
}; };
const handleStyleChange = (style) => { const handleStyleChange = (style)=>{
setCurrentBrew(prevBrew => ({ ...prevBrew, style })); setCurrentBrew((prevBrew)=>({ ...prevBrew, style }));
}; };
const handleSnipChange = (snippet)=>{ const handleSnipChange = (snippet)=>{
//If there are HTML errors, run the validator on every change to give quick feedback //If there are HTML errors, run the validator on every change to give quick feedback
if(HTMLErrors.length) if(HTMLErrors.length)
setHTMLErrors(Markdown.validate(snippet)); setHTMLErrors(Markdown.validate(snippet));
setCurrentBrew((prevBrew) => ({ ...prevBrew, snippets: snippet })); setCurrentBrew((prevBrew)=>({ ...prevBrew, snippets: snippet }));
}; };
const handleMetaChange = (metadata, field = undefined) => { const handleMetaChange = (metadata, field = undefined)=>{
if (field === 'theme' || field === 'renderer') if(field === 'theme' || field === 'renderer')
fetchThemeBundle(setError, setThemeBundle, metadata.renderer, metadata.theme); fetchThemeBundle(setError, setThemeBundle, metadata.renderer, metadata.theme);
setCurrentBrew(prev => ({ ...prev, ...metadata })); setCurrentBrew((prev)=>({ ...prev, ...metadata }));
}; };
const updateBrew = (newData) => const updateBrew = (newData)=>setCurrentBrew((prevBrew)=>({
setCurrentBrew((prevBrew) => ({ ...prevBrew,
...prevBrew, style : newData.style,
style : newData.style, text : newData.text,
text : newData.text, snippets : newData.snippets
snippets : newData.snippets }));
}));
const trySave = (immediate = false)=>{
const trySave = (immediate = false) => {
//debounceSave = _.debounce(save, SAVE_TIMEOUT); //debounceSave = _.debounce(save, SAVE_TIMEOUT);
if (isSaving) return; if(isSaving) return;
const hasChange = !_.isEqual(currentBrew, savedBrew.current); const hasChange = !_.isEqual(currentBrew, savedBrew.current);
if (immediate) { if(immediate) {
debounceSave(currentBrew, saveGoogle); debounceSave(currentBrew, saveGoogle);
debounceSave.flush?.(); debounceSave.flush?.();
return; return;
} }
if (hasChange) if(hasChange)
debounceSave(currentBrew, saveGoogle); debounceSave(currentBrew, saveGoogle);
else else
debounceSave.cancel?.(); debounceSave.cancel?.();
}; };
const handleGoogleClick = () => { const handleGoogleClick = ()=>{
if (!global.account?.googleId) { if(!global.account?.googleId) {
setAlertLoginToTransfer(true); setAlertLoginToTransfer(true);
return; return;
} }
setConfirmGoogleTransfer((prev) => !prev); setConfirmGoogleTransfer((prev)=>!prev);
setError(null); setError(null);
}; };
const closeAlerts = (e) => { const closeAlerts = (e)=>{
e.stopPropagation(); //Only handle click once so alert doesn't reopen e.stopPropagation(); //Only handle click once so alert doesn't reopen
setAlertTrashedGoogleBrew(false); setAlertTrashedGoogleBrew(false);
setAlertLoginToTransfer(false); setAlertLoginToTransfer(false);
setConfirmGoogleTransfer(false); setConfirmGoogleTransfer(false);
}; };
const toggleGoogleStorage = () => { const toggleGoogleStorage = ()=>{
setSaveGoogle((prev) => !prev); setSaveGoogle((prev)=>!prev);
setError(null); setError(null);
trySave(true); trySave(true);
}; };
const save = async (brew, saveToGoogle) => { const save = async (brew, saveToGoogle)=>{
debounceSave?.cancel?.(); debounceSave?.cancel?.();
setIsSaving(true); setIsSaving(true);
@@ -211,11 +208,11 @@ const EditPage = (props) => {
//Prepare content to send to server //Prepare content to send to server
const brewToSave = { const brewToSave = {
...brew, ...brew,
text : brew.text.normalize('NFC'), text : brew.text.normalize('NFC'),
pageCount: ((brew.renderer === 'legacy' ? brew.text.match(/\\page/g) : brew.text.match(/^\\page$/gm)) || []).length + 1, pageCount : ((brew.renderer === 'legacy' ? brew.text.match(/\\page/g) : brew.text.match(/^\\page$/gm)) || []).length + 1,
patches : stringifyPatches(makePatches(encodeURI(savedBrew.current.text.normalize('NFC')), encodeURI(brew.text.normalize('NFC')))), patches : stringifyPatches(makePatches(encodeURI(savedBrew.current.text.normalize('NFC')), encodeURI(brew.text.normalize('NFC')))),
hash : await md5(savedBrew.current.text), hash : await md5(savedBrew.current.text),
textBin : undefined textBin : undefined
}; };
const compressedBrew = gzipSync(strToU8(JSON.stringify(brewToSave))); const compressedBrew = gzipSync(strToU8(JSON.stringify(brewToSave)));
@@ -237,7 +234,7 @@ const EditPage = (props) => {
savedBrew.current = { savedBrew.current = {
...brew, ...brew,
googleId: googleId ?? null, googleId : googleId ?? null,
editId, editId,
shareId, shareId,
version version
@@ -251,7 +248,7 @@ const EditPage = (props) => {
history.replaceState(null, null, `/edit/${editId}`); history.replaceState(null, null, `/edit/${editId}`);
}; };
const renderGoogleDriveIcon = () => ( const renderGoogleDriveIcon = ()=>(
<Nav.item className='googleDriveStorage' onClick={handleGoogleClick}> <Nav.item className='googleDriveStorage' onClick={handleGoogleClick}>
<img src={googleDriveIcon} className={saveGoogle ? '' : 'inactive'} alt='Google Drive icon' /> <img src={googleDriveIcon} className={saveGoogle ? '' : 'inactive'} alt='Google Drive icon' />
@@ -286,13 +283,13 @@ const EditPage = (props) => {
</Nav.item> </Nav.item>
); );
const renderSaveButton = () => { const renderSaveButton = ()=>{
// #1 - Currently saving, show SAVING // #1 - Currently saving, show SAVING
if (isSaving) if(isSaving)
return <Nav.item className='save' icon='fas fa-spinner fa-spin'>saving...</Nav.item>; return <Nav.item className='save' icon='fas fa-spinner fa-spin'>saving...</Nav.item>;
// #2 - Unsaved changes exist, autosave is OFF and warning timer has expired, show AUTOSAVE WARNING // #2 - Unsaved changes exist, autosave is OFF and warning timer has expired, show AUTOSAVE WARNING
if (unsavedChanges && autoSaveWarning) { if(unsavedChanges && autoSaveWarning) {
resetAutoSaveWarning(); resetAutoSaveWarning();
const elapsedTime = Math.round((new Date() - unsavedTime) / 1000 / 60); const elapsedTime = Math.round((new Date() - unsavedTime) / 1000 / 60);
const text = elapsedTime === 0 const text = elapsedTime === 0
@@ -306,42 +303,42 @@ const EditPage = (props) => {
} }
// #3 - Unsaved changes exist, click to save, show SAVE NOW // #3 - Unsaved changes exist, click to save, show SAVE NOW
if (unsavedChanges) if(unsavedChanges)
return <Nav.item className='save' onClick={() => trySave(true)} color='blue' icon='fas fa-save'>Save Now</Nav.item> return <Nav.item className='save' onClick={()=>trySave(true)} color='blue' icon='fas fa-save'>Save Now</Nav.item>;
// #4 - No unsaved changes, autosave is ON, show AUTO-SAVED // #4 - No unsaved changes, autosave is ON, show AUTO-SAVED
if (autoSaveEnabled) if(autoSaveEnabled)
return <Nav.item className='save saved'>auto-saved.</Nav.item>; return <Nav.item className='save saved'>auto-saved.</Nav.item>;
// DEFAULT - No unsaved changes, show SAVED // DEFAULT - No unsaved changes, show SAVED
return <Nav.item className='save saved'>saved.</Nav.item>; return <Nav.item className='save saved'>saved.</Nav.item>;
}; };
const toggleAutoSave = () => { const toggleAutoSave = ()=>{
if (warningTimer.current) clearTimeout(warningTimer.current); if(warningTimer.current) clearTimeout(warningTimer.current);
localStorage.setItem('AUTOSAVE_ON', JSON.stringify(!autoSaveEnabled)); localStorage.setItem('AUTOSAVE_ON', JSON.stringify(!autoSaveEnabled));
setAutoSaveWarning(autoSaveWarning); setAutoSaveWarning(autoSaveWarning);
setAutoSaveEnabled(!autoSaveEnabled); setAutoSaveEnabled(!autoSaveEnabled);
}; };
const resetAutoSaveWarning = () => { const resetAutoSaveWarning = ()=>{
setTimeout(()=>setAutoSaveWarning(false), UNSAVED_WARNING_POPUP_TIMEOUT); // Hide the warning after 4 seconds setTimeout(()=>setAutoSaveWarning(false), UNSAVED_WARNING_POPUP_TIMEOUT); // Hide the warning after 4 seconds
warningTimer.current = setTimeout(()=>setAutoSaveWarning(true), UNSAVED_WARNING_TIMEOUT); // 15 minutes between unsaved changes warnings warningTimer.current = setTimeout(()=>setAutoSaveWarning(true), UNSAVED_WARNING_TIMEOUT); // 15 minutes between unsaved changes warnings
}; };
const renderAutoSaveButton = () => ( const renderAutoSaveButton = ()=>(
<Nav.item onClick={toggleAutoSave}> <Nav.item onClick={toggleAutoSave}>
Autosave <i className={autoSaveEnabled ? 'fas fa-power-off active' : 'fas fa-power-off'}></i> Autosave <i className={autoSaveEnabled ? 'fas fa-power-off active' : 'fas fa-power-off'}></i>
</Nav.item> </Nav.item>
); );
const processShareId = () => ( const processShareId = ()=>(
currentBrew.googleId && !currentBrew.stubbed currentBrew.googleId && !currentBrew.stubbed
? currentBrew.googleId + currentBrew.shareId ? currentBrew.googleId + currentBrew.shareId
: currentBrew.shareId : currentBrew.shareId
); );
const getRedditLink = () => { const getRedditLink = ()=>{
const shareLink = processShareId(); const shareLink = processShareId();
const systems = currentBrew.systems.length > 0 ? ` [${currentBrew.systems.join(' - ')}]` : ''; const systems = currentBrew.systems.length > 0 ? ` [${currentBrew.systems.join(' - ')}]` : '';
const title = `${currentBrew.title} ${systems}`; const title = `${currentBrew.title} ${systems}`;
@@ -352,7 +349,7 @@ const EditPage = (props) => {
return `https://www.reddit.com/r/UnearthedArcana/submit?title=${encodeURIComponent(title.toWellFormed())}&text=${encodeURIComponent(text)}`; return `https://www.reddit.com/r/UnearthedArcana/submit?title=${encodeURIComponent(title.toWellFormed())}&text=${encodeURIComponent(text)}`;
}; };
const clearError = () => { const clearError = ()=>{
setError(null); setError(null);
setIsSaving(false); setIsSaving(false);
}; };