Make useLocalStorage actually work

This commit is contained in:
Gwilym Inzani 2023-07-05 11:59:32 +01:00 committed by Corwin
parent 472d641cfc
commit 683afbf133
No known key found for this signature in database
3 changed files with 138 additions and 118 deletions

View file

@ -1,5 +1,5 @@
import React, { useState } from "react"; import { useRef, useState } from "react";
import { Mgba } from "./mgba"; import { Mgba, MgbaHandle } from "./mgba";
import { BindingsControl, DefaultBindingsSet, Bindings } from "./bindings"; import { BindingsControl, DefaultBindingsSet, Bindings } from "./bindings";
import { styled } from "styled-components"; import { styled } from "styled-components";
import { useOnKeyUp } from "./useOnKeyUp.hook"; import { useOnKeyUp } from "./useOnKeyUp.hook";
@ -16,24 +16,28 @@ const VolumeLabel = styled.label`
margin-bottom: 20px; margin-bottom: 20px;
`; `;
const CloseButton = styled.button` const ActionButton = styled.button`
width: 100%; width: 100%;
margin-top: 20px; margin-top: 20px;
`; `;
function App() { function App() {
const [volumeState, setVolume] = useState(1.0); const [{ volume, bindings }, setState] = useLocalStorage(
const [bindingsState, setBindings] = useState(DefaultBindingsSet()); { volume: 1.0, bindings: DefaultBindingsSet() },
const { volume, bindings } = useLocalStorage(
{ volume: volumeState, bindings: bindingsState },
"agbrswebplayer" "agbrswebplayer"
); );
const setVolume = (newVolume: number) =>
setState({ volume: newVolume, bindings });
const setBindings = (newBindings: Bindings) =>
setState({ volume, bindings: newBindings });
const [paused, setPaused] = useState(false); const [paused, setPaused] = useState(false);
const [showBindings, setShowBindings] = useState(false); const [showBindings, setShowBindings] = useState(false);
const mgbaRef = useRef<MgbaHandle>(null);
useOnKeyUp("Escape", () => { useOnKeyUp("Escape", () => {
setShowBindings(!showBindings); setShowBindings(!showBindings);
}); });
@ -48,9 +52,11 @@ function App() {
volume={volume} volume={volume}
setVolume={setVolume} setVolume={setVolume}
hide={() => setShowBindings(false)} hide={() => setShowBindings(false)}
restart={() => mgbaRef.current?.restart()}
/> />
)} )}
<Mgba <Mgba
ref={mgbaRef}
gameUrl="/game.gba" gameUrl="/game.gba"
volume={volume} volume={volume}
controls={bindings.Actual} controls={bindings.Actual}
@ -67,6 +73,7 @@ function BindingsWindow({
volume, volume,
setVolume, setVolume,
hide, hide,
restart,
}: { }: {
bindings: Bindings; bindings: Bindings;
setBindings: (b: Bindings) => void; setBindings: (b: Bindings) => void;
@ -74,6 +81,7 @@ function BindingsWindow({
volume: number; volume: number;
setVolume: (v: number) => void; setVolume: (v: number) => void;
hide: () => void; hide: () => void;
restart: () => void;
}) { }) {
return ( return (
<BindingsDialog open onClose={hide}> <BindingsDialog open onClose={hide}>
@ -97,7 +105,8 @@ function BindingsWindow({
setBindings={setBindings} setBindings={setBindings}
setPaused={setPaused} setPaused={setPaused}
/> />
<CloseButton onClick={hide}>Close</CloseButton> <ActionButton onClick={restart}>Restart</ActionButton>
<ActionButton onClick={hide}>Close</ActionButton>
</BindingsDialog> </BindingsDialog>
); );
} }

View file

@ -1,4 +1,11 @@
import { FC, useEffect, useRef, useState } from "react"; import {
FC,
forwardRef,
useEffect,
useImperativeHandle,
useRef,
useState,
} from "react";
import mGBA from "./vendor/mgba"; import mGBA from "./vendor/mgba";
import { KeyBindings } from "./bindings"; import { KeyBindings } from "./bindings";
import { styled } from "styled-components"; import { styled } from "styled-components";
@ -28,7 +35,12 @@ const MgbaCanvas = styled.canvas`
max-height: 100%; max-height: 100%;
`; `;
export const Mgba: FC<MgbaProps> = ({ gameUrl, volume, controls, paused }) => { export interface MgbaHandle {
restart: () => void;
}
export const Mgba = forwardRef<MgbaHandle, MgbaProps>(
({ gameUrl, volume, controls, paused }, ref) => {
const canvas = useRef(null); const canvas = useRef(null);
const mgbaModule = useRef<Module>({}); const mgbaModule = useRef<Module>({});
@ -112,5 +124,12 @@ export const Mgba: FC<MgbaProps> = ({ gameUrl, volume, controls, paused }) => {
} }
}, [gameLoaded, paused]); }, [gameLoaded, paused]);
useImperativeHandle(ref, () => {
return {
restart: () => mgbaModule.current.quickReload(),
};
});
return <MgbaCanvas ref={canvas} />; return <MgbaCanvas ref={canvas} />;
}; }
);

View file

@ -1,36 +1,28 @@
import { useRef, useLayoutEffect, useEffect } from "react"; import { useCallback, useEffect, useState } from "react";
export const useLocalStorage = <T>(currentValue: T, appName: string): T => { export const useLocalStorage = <T>(
const initialValue = useRef<T>(); defaultValue: T,
appName: string
const isFirstRun = !initialValue.current; ): [T, (newValue: T) => void] => {
const [value, setValue] = useState(() => {
useLayoutEffect(() => {
if (!initialValue.current) {
try { try {
const storageValue = localStorage.getItem(appName); const storageValue = localStorage.getItem(appName);
if (storageValue) { if (storageValue) {
initialValue.current = JSON.parse(storageValue); return JSON.parse(storageValue);
} else { } else {
initialValue.current = currentValue; return defaultValue;
} }
} catch { } catch {
initialValue.current = currentValue; return defaultValue;
}
} }
});
const setStoredValue = useCallback((newValue: T) => {
setValue(newValue);
try {
localStorage.setItem(appName, JSON.stringify(newValue));
} catch {}
}, []); }, []);
useEffect(() => { return [value, setStoredValue];
try {
if (initialValue.current && currentValue) {
localStorage.setItem(appName, JSON.stringify(currentValue));
}
} catch {}
}, [currentValue]);
if (isFirstRun) {
return initialValue.current ?? currentValue;
} else {
return currentValue;
}
}; };