为了陷入React钩子,我决定尝试制作蛇并使用useReducer / useContext状态管理。
现在,我被封锁在想要定向按键更改活动图块状态的位置。在useReducer中,这似乎接收到正确的有效负载,并创建了正确的有效负载对象,但是当它更新状态时,它是未定义的?
按下向下键将显示错误消息“ TypeError:无法读取未定义的属性'some'”。
Board.jsx
import React, { useContext, useEffect } from "react";
import Tile from "./Tile.jsx";
import { snakeContext } from '../contexts/snakeContext'
const Board = () => {
const {
state: {
snake, food, direction, gameOver
},
dispatch,
rows,
cols
} = useContext(snakeContext)
useEffect(() => {
const onKeyPress = (e) => {
switch (e.keyCode) {
case 38: //Up
return direction === "down" || dispatch({ type: 'DIRECTION', payload: "up" });
case 40: // Down
return direction === "up" || dispatch({ type: 'DIRECTION', payload: "down" });
case 37: //Left
return direction === "right" || dispatch({ type: 'DIRECTION', payload: "left" });
case 39: // Right
return direction === "left" ||
dispatch({ type: 'DIRECTION', payload: "right" });
default:
break;
}
};
window.addEventListener("keydown", onKeyPress);
return () => window.removeEventListener("keydown", onKeyPress);
}, [direction]);
useEffect(() => {
const interval = setInterval(() => {
switch (direction) {
case "up":
dispatch({ type: 'SNAKE', payload: { ...snake[0], y: snake[0].y - 1 } })
break
case "down":
dispatch({ type: 'SNAKE', payload: { ...snake[0], y: snake[0].y + 1 } })
break;
case "left":
dispatch({ type: 'SNAKE', payload: { ...snake[0], x: snake[0].x - 1 } })
break;
case "right":
dispatch({ type: 'SNAKE', payload: { ...snake[0], x: snake[0].x + 1 } })
break;
default:
break;
}
}, 500);
return () => clearInterval(interval);
});
const style = {
maxHeight: `${2 * rows}rem`,
maxWidth: `${2 * cols}rem`,
margin: "0 auto",
paddingTop: "4rem"
};
const isActiveMatchingState = (i, j) => {
return snake.some(snakeTile =>
snakeTile.y === i && snakeTile.x === j
)
}
const renderBoard = () => {
let grid = Array.from(Array(rows), () => new Array(cols));
for (let i = 0; i < grid.length; i++) {
for (let j = 0; j < grid[i].length; j++) {
grid[i][j] = (
<Tile
isActive={isActiveMatchingState(i, j)}
isFood={food.y === i && food.x === j}
key={`${[i, j]}`}
/>
);
}
}
return grid;
};
return (
gameOver ?
<div>GAME OVER</div> :
<div style={style}>{renderBoard()}</div>
)
};
export default Board;
snakeReducer.jsx
export const snakeReducer = (state, action) => {
const { type, payload } = action;
switch (type) {
case 'SNAKE':
return [{ ...state.snake[0], x: payload.x, y: payload.y }]
case 'FOOD':
return { ...state, x: payload.x, y: payload.y };
case 'DIRECTION':
return { ...state, direction: payload };
case 'GAME_OVER':
return { ...state, gameOver: payload };
default:
throw new Error();
}
};
我的useContext设置按照建议使用useMemo-https://hswolff.com/blog/how-to-usecontext-with-usereducer/
snakeContext.js
import React, { createContext, useReducer, useMemo } from 'react';
import { snakeReducer } from '../reducers/snakeReducer';
export const snakeContext = createContext();
const rows = 20;
const cols = 15;
const randomPosition = (biggestNumber) => Math.floor(Math.random() * biggestNumber)
const initialState = {
snake: [{ x: 0, y: 0 }],
food: { x: randomPosition(rows), y: randomPosition(cols) },
direction: null,
gameOver: false
};
const SnakeContextProvider = ({ children }) => {
const [state, dispatch] = useReducer(snakeReducer, initialState);
const contextValue = useMemo(() => ({ state, rows, cols, dispatch }), [state, dispatch]);
return <snakeContext.Provider value={contextValue}>{children}</snakeContext.Provider>;
};
export default SnakeContextProvider;
App.js
import React from 'react';
import Home from './pages/Home';
import SnakeContextProvider from './contexts/snakeContext';
import './App.css';
const App = () => {
return (
<SnakeContextProvider>
<Home />
</SnakeContextProvider>
)
};
export default App;
Home.jsx是包含Board.jsx的页面组件
奇怪的是,方向键上的更新可以正常更新,因此useReducer似乎设置正确。
当前完整的回购在这里-https://github.com/puyanwei/snake
谢谢!
答案 0 :(得分:0)
请您更新一下吗?
export const snakeReducer = (state, action) => {
const { type, payload } = action;
switch (type) {
case 'SNAKE':
return [{ ...state.snake[0], x: payload.x, y: payload.y }]
case 'FOOD':
return { ...state, x: payload.x, y: payload.y };
case 'DIRECTION':
return { ...state, direction: payload };
case 'GAME_OVER':
return { ...state, gameOver: payload };
default:
return state; //error here: throw new Error();
}
};
答案 1 :(得分:0)
在化简器中对SNAKE
动作的处理似乎不正确。您要返回一个数组,但可能期望的是像初始状态一样的状态,对吗?
const initialState = {
snake: [{ x: 0, y: 0 }],
prev: { x: null, y: null },
food: { x: randomPosition(rows), y: randomPosition(cols) },
direction: null,
gameOver: false
};
由于SNAKE
为snake[0]
,{ x:..., y: ...}
操作的化简器的返回值与此类似:
[{ x: payload.x, y: payload.y }]
答案 2 :(得分:0)
最后是我的减速器,“蛇”的正确回报应该是;
case 'SNAKE':
return {
...state,
snake: [{ x: payload.x, y: payload.y }, ...state.snake]
};
感谢所有帮助过的人!