Created custom implemtation of atomWithStorage, in order to fix undefined on first render
This commit is contained in:
parent
aeba6343e3
commit
800ba32064
@ -1,6 +1,6 @@
|
|||||||
import React, {FC} from "react";
|
import React, {FC} from "react";
|
||||||
import {useAtom, useAtomValue} from "jotai";
|
import {useAtom, useAtomValue} from "jotai";
|
||||||
import {selectedDiceAtom, thisPlayerAtom} from "../utils/state";
|
import {getPlayerAtom, selectedDiceAtom,} from "../utils/state";
|
||||||
import {Button} from "./button";
|
import {Button} from "./button";
|
||||||
|
|
||||||
export const AllDice: FC<{ values?: number[] } & ComponentProps> = (
|
export const AllDice: FC<{ values?: number[] } & ComponentProps> = (
|
||||||
@ -38,7 +38,7 @@ export const Dice: FC<DiceProps> = (
|
|||||||
onClick,
|
onClick,
|
||||||
}) => {
|
}) => {
|
||||||
|
|
||||||
const thisPlayer = useAtomValue(thisPlayerAtom);
|
const thisPlayer = useAtomValue(getPlayerAtom);
|
||||||
|
|
||||||
function handleClick() {
|
function handleClick() {
|
||||||
if (onClick && value) {
|
if (onClick && value) {
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import React, {FC, MouseEventHandler} from "react";
|
import React, {FC, MouseEventHandler} from "react";
|
||||||
import {State} from "../game/player";
|
import {State} from "../game/player";
|
||||||
import {currentPlayerAtom, playersAtom, rollDiceButtonAtom, thisPlayerAtom} from "../utils/state";
|
import {currentPlayerAtom, getPlayerAtom, playersAtom, rollDiceButtonAtom} from "../utils/state";
|
||||||
import {useAtomValue} from "jotai";
|
import {useAtomValue} from "jotai";
|
||||||
import {Button} from "./button";
|
import {Button} from "./button";
|
||||||
import rules from "../game/rules";
|
import rules from "../game/rules";
|
||||||
@ -17,7 +17,7 @@ const GameButton: FC<GameButtonProps> = (
|
|||||||
}) => {
|
}) => {
|
||||||
|
|
||||||
const currentPlayer = useAtomValue(currentPlayerAtom);
|
const currentPlayer = useAtomValue(currentPlayerAtom);
|
||||||
const thisPlayer = useAtomValue(thisPlayerAtom);
|
const thisPlayer = useAtomValue(getPlayerAtom);
|
||||||
const players = useAtomValue(playersAtom);
|
const players = useAtomValue(playersAtom);
|
||||||
const activeRollDiceButton = useAtomValue(rollDiceButtonAtom);
|
const activeRollDiceButton = useAtomValue(rollDiceButtonAtom);
|
||||||
|
|
||||||
|
@ -1,24 +1,22 @@
|
|||||||
import React, {FC, useEffect} from "react";
|
import React, {FC, useEffect} from "react";
|
||||||
import {GameComponent} from "../components/gameComponent";
|
import {GameComponent} from "../components/gameComponent";
|
||||||
import {useAtomValue} from "jotai";
|
import {useAtomValue} from "jotai";
|
||||||
import {selectedMapAtom, thisPlayerAtom} from "../utils/state";
|
import {getPlayerAtom, selectedMapAtom} from "../utils/state";
|
||||||
|
|
||||||
const Game: FC = () => { // TODO gameId in path
|
const Game: FC = () => {
|
||||||
const player = useAtomValue(thisPlayerAtom);
|
const player = useAtomValue(getPlayerAtom);
|
||||||
const map = useAtomValue(selectedMapAtom);
|
const map = useAtomValue(selectedMapAtom);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
console.debug(player);
|
|
||||||
if (!player) {
|
if (!player) {
|
||||||
// TODO player is undefined on first render, then defined on second render
|
window.location.href = "/";
|
||||||
// window.location.href = "/";
|
|
||||||
}
|
}
|
||||||
}, [player]);
|
}, []);
|
||||||
|
|
||||||
if (player && map) {
|
if (player && map) {
|
||||||
return <GameComponent player={player} map={map}/>;
|
return <GameComponent player={player} map={map}/>;
|
||||||
} else {
|
} else {
|
||||||
throw new Error("Player or map is undefined");
|
return null;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -5,13 +5,13 @@ import Dropdown from "../components/dropdown";
|
|||||||
import {Colour, getColours} from "../game/colour";
|
import {Colour, getColours} from "../game/colour";
|
||||||
import {useNavigate} from "react-router-dom";
|
import {useNavigate} from "react-router-dom";
|
||||||
import {useSetAtom} from "jotai";
|
import {useSetAtom} from "jotai";
|
||||||
import {thisPlayerAtom} from "../utils/state";
|
import {setPlayerAtom} from "../utils/state";
|
||||||
|
|
||||||
const Home: FC = () => {
|
const Home: FC = () => {
|
||||||
|
|
||||||
const input = useRef<HTMLInputElement>(null);
|
const input = useRef<HTMLInputElement>(null);
|
||||||
const dropdown = useRef<HTMLSelectElement>(null);
|
const dropdown = useRef<HTMLSelectElement>(null);
|
||||||
const setPlayer = useSetAtom(thisPlayerAtom);
|
const setPlayer = useSetAtom(setPlayerAtom);
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
function formHandler(): void {
|
function formHandler(): void {
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import React, {FC, Suspense} from "react";
|
import React, {FC, Suspense, useEffect} from "react";
|
||||||
import {atom, useAtomValue} from "jotai";
|
import {atom, useAtomValue} from "jotai";
|
||||||
import {Button} from "../components/button";
|
import {Button} from "../components/button";
|
||||||
import {selectedMapAtom, thisPlayerAtom} from "../utils/state";
|
import {getPlayerAtom, selectedMapAtom} from "../utils/state";
|
||||||
import {getData, postData} from "../utils/api";
|
import {getData, postData} from "../utils/api";
|
||||||
import {getPacManSpawns} from "../game/map";
|
import {getPacManSpawns} from "../game/map";
|
||||||
import {useNavigate} from "react-router-dom";
|
import {useNavigate} from "react-router-dom";
|
||||||
@ -13,7 +13,7 @@ const fetchAtom = atom(async () => {
|
|||||||
|
|
||||||
const LobbyPage: FC = () => { // TODO check if player is defined in storage, if not redirect to login
|
const LobbyPage: FC = () => { // TODO check if player is defined in storage, if not redirect to login
|
||||||
|
|
||||||
const thisPlayer = useAtomValue(thisPlayerAtom);
|
const thisPlayer = useAtomValue(getPlayerAtom);
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
const map = useAtomValue(selectedMapAtom);
|
const map = useAtomValue(selectedMapAtom);
|
||||||
|
|
||||||
@ -35,6 +35,10 @@ const LobbyPage: FC = () => { // TODO check if player is defined in storage, if
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
console.debug(thisPlayer)
|
||||||
|
})
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Button onClick={createGame}>New game</Button>
|
<Button onClick={createGame}>New game</Button>
|
||||||
@ -50,7 +54,7 @@ export default LobbyPage;
|
|||||||
const GameTable: FC<ComponentProps> = ({className}) => {
|
const GameTable: FC<ComponentProps> = ({className}) => {
|
||||||
|
|
||||||
const data = useAtomValue(fetchAtom);
|
const data = useAtomValue(fetchAtom);
|
||||||
const thisPlayer = useAtomValue(thisPlayerAtom);
|
const thisPlayer = useAtomValue(getPlayerAtom);
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
async function joinGame(gameId: string): Promise<void> {
|
async function joinGame(gameId: string): Promise<void> {
|
||||||
|
@ -2,14 +2,14 @@ import React, {FC, FormEvent} from "react";
|
|||||||
import {Button} from "../components/button";
|
import {Button} from "../components/button";
|
||||||
import Input from "../components/input";
|
import Input from "../components/input";
|
||||||
import {useSetAtom} from "jotai";
|
import {useSetAtom} from "jotai";
|
||||||
import {thisPlayerAtom} from "../utils/state";
|
import {setPlayerAtom} from "../utils/state";
|
||||||
import Player from "../game/player";
|
import Player from "../game/player";
|
||||||
import {useNavigate} from "react-router-dom";
|
import {useNavigate} from "react-router-dom";
|
||||||
import {postData} from "../utils/api";
|
import {postData} from "../utils/api";
|
||||||
|
|
||||||
const Login: FC = () => {
|
const Login: FC = () => {
|
||||||
|
|
||||||
const setThisPlayer = useSetAtom(thisPlayerAtom);
|
const setThisPlayer = useSetAtom(setPlayerAtom);
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
async function handleLogin(e: FormEvent<HTMLFormElement>): Promise<void> {
|
async function handleLogin(e: FormEvent<HTMLFormElement>): Promise<void> {
|
||||||
|
@ -1,10 +1,9 @@
|
|||||||
import Player from "../game/player";
|
import Player from "../game/player";
|
||||||
import {atom} from "jotai";
|
import {atom} from "jotai";
|
||||||
import {atomWithStorage, createJSONStorage} from "jotai/utils";
|
|
||||||
import {Ghost} from "../game/character";
|
import {Ghost} from "../game/character";
|
||||||
import {customMap} from "../game/map";
|
import {customMap} from "../game/map";
|
||||||
|
|
||||||
const playerStorage = createJSONStorage<Player | undefined>(() => sessionStorage);
|
const playerStorage = "player";
|
||||||
/**
|
/**
|
||||||
* All players in the game.
|
* All players in the game.
|
||||||
*/
|
*/
|
||||||
@ -22,14 +21,35 @@ export const ghostsAtom = atom<Ghost[]>([]);
|
|||||||
*/
|
*/
|
||||||
export const allCharactersAtom = atom(get => [...get(playerCharactersAtom), ...get(ghostsAtom)]);
|
export const allCharactersAtom = atom(get => [...get(playerCharactersAtom), ...get(ghostsAtom)]);
|
||||||
/**
|
/**
|
||||||
* The player that is currently using this browser.
|
* The player that is currently logged in.
|
||||||
*/
|
*/
|
||||||
export const thisPlayerAtom = atomWithStorage<Player | undefined>("player", undefined, {
|
const playerAtom = atom<Player | undefined>(undefined);
|
||||||
...playerStorage,
|
/**
|
||||||
getItem(key, initialValue): Player | undefined {
|
* Gets the player that is currently logged in. If playerAtom is undefined, then it will try to get the player from session storage.
|
||||||
const playerProps = playerStorage.getItem(key, initialValue) as PlayerProps | undefined;
|
* @returns An atom representing the player that is currently logged in, or undefined if there is no player logged in.
|
||||||
return playerProps ? new Player(playerProps) : undefined;
|
*/
|
||||||
},
|
export const getPlayerAtom = atom(get => {
|
||||||
|
const atomValue = get(playerAtom);
|
||||||
|
if (!atomValue) {
|
||||||
|
const item = sessionStorage.getItem(playerStorage);
|
||||||
|
if (item) {
|
||||||
|
const playerProps = JSON.parse(item) as PlayerProps;
|
||||||
|
return new Player(playerProps);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return atomValue;
|
||||||
|
});
|
||||||
|
/**
|
||||||
|
* Sets the player that is currently logged in. If player is undefined, then it will remove the player from session storage.
|
||||||
|
* @param player The player that is currently logged in, or undefined if there is no player logged in.
|
||||||
|
* @return An atom used to set the player that is currently logged in.
|
||||||
|
*/
|
||||||
|
export const setPlayerAtom = atom(null, (get, set, player: Player | undefined) => {
|
||||||
|
set(playerAtom, player);
|
||||||
|
if (player)
|
||||||
|
sessionStorage.setItem(playerStorage, JSON.stringify(player));
|
||||||
|
else
|
||||||
|
sessionStorage.removeItem(playerStorage);
|
||||||
});
|
});
|
||||||
/**
|
/**
|
||||||
* All dice that have been rolled.
|
* All dice that have been rolled.
|
||||||
|
Loading…
x
Reference in New Issue
Block a user