Не могу получить доступ к моему состоянию в моей функции requestAnimationFrame - PullRequest
0 голосов
/ 02 ноября 2019

Я пытаюсь построить игру космического захватчика с нуля, используя React / React-Hook и холст HTML5.

Пока мне удалось нарисовать свой корабль на холсте, но я не могу понять, какчтобы получить доступ к моим состояниям в функции requestAnimationFrame. Мне удалось получить доступ к ссылкам, но я не хочу, чтобы все мои переменные были ссылками.

Пока мой код выглядит так:

import React from 'react';
import {makeStyles} from '@material-ui/core/styles';

const spaceInvaderStyles = makeStyles((theme) => ({
  canvas: {
    display: 'block',
    margin: 'auto',
    imageRendering: 'optimizeSpeed',
    imageRendering: '-moz-crisp-edges',
    imageRendering: '-webkit-optimize-contrast',
    imageRendering: 'optimize-contrast',
    backgroundColor: 'black',
  }
}))



// GAME CONSTANTS
const GAME = {
  shape: {
    w:'640px',
    h:'640px',
  },
  shipRow: '600',
  shipColor: 'rgba(0,252,0)',
  spritesSrc: '',
  sprites: {
    ship: {
      x: 0,
      y: 204,
      w: 61,
      h: 237,
    }
  },
  player: {
    initialPos: {
      x: 290,
      y: 580,
    }
  }


}

const KEYS = {
  left:81,
  right:68,
  down: 83,
  up: 90,
  arrowLeft: 37,
  arrowRight: 39,
  arrowDown: 40,
  arrowUp: 38,
}



const SpaceInvader = (props) => {

  const classes = spaceInvaderStyles();

  const canvasRef = React.useRef();

  const [cctx, setCctx] = React.useState(null);
  const [sprites, setSprites] = React.useState(null);
  const [player, setPlayer] = React.useState({
    pos: GAME.player.initialPos.x,
  })

  // keys
  const [currentKey, setCurrentKey] = React.useState(null);

  //time
  let lastTime = 0;

  const [counter, setCounter] = React.useState(0);


  React.useEffect(() => {

    // INIT

    // context
    const canvas = canvasRef.current;
    const context = canvas.getContext('2d');
    setCctx(context)

    // sprites
    loadSprites();

  }, [])

  // key handler
  React.useEffect(() => {
    window.addEventListener('keydown', (event) => handleUserKeyPress(event,true));
    window.addEventListener('keyup', (event) => handleUserKeyPress(event,false))

    return () => {
      window.removeEventListener('keydown', (event) => handleUserKeyPress(event,true));
      window.removeEventListener('keyup', (event) => handleUserKeyPress(event,false))
    };
  }, [cctx, sprites, player, currentKey])

  React.useEffect(() => {
    if(!cctx) return
    animate();
  }, [cctx])

  React.useEffect(() => {

    if(spritesAreLoaded()){
      cctx.drawImage(sprites, GAME.sprites.ship.x, GAME.sprites.ship.y, GAME.sprites.ship.w, GAME.sprites.ship.h, GAME.player.initialPos.x, GAME.player.initialPos.y , GAME.sprites.ship.w, GAME.sprites.ship.h)

    }
  }, [sprites])

  React.useEffect(() => {
    console.log(counter)
  }, [counter])

  // utils
  const clearCanvas = () => {

    cctx.clearRect(0,0, 640, 640);
  }

  const saveCanvas = () => {
    cctx.save();
  }

  const drawImage = (image, sx, sy, sw, dx, dy, sh, dw, dh) => {
    cctx.drawImage(image, sx, sy, sw, dx, dy, sh, dw, dh);
  }

  const restore = () => {
    cctx.restore();
  }

  const loadSprites = () => {
    var spritesImg = new Image();
    spritesImg.src = GAME.spritesSrc;
    spritesImg.onload = function() {
      // sprites are loaded at this point
      setSprites(spritesImg);
    }

  }

  const spritesAreLoaded = () => {
    return sprites !== null;
  }

  const move = (direction) => {
    // cctx, sprites and all others state vars are at default value here too
    clearCanvas();
    saveCanvas();
    drawImage(sprites, GAME.sprites.ship.x, GAME.sprites.ship.y, GAME.sprites.ship.w, GAME.sprites.ship.h, player.pos + (10 * direction), GAME.player.initialPos.y , GAME.sprites.ship.w, GAME.sprites.ship.h);
    restore();
    setPlayer({...player, pos: player.pos + (10 * direction)});
  }



  const handleUserKeyPress = React.useCallback( (event, isDown) => {

    event.preventDefault();
    const {key, keyCode} = event;
    setCurrentKey(isDown ? keyCode : null);

  }, [cctx, sprites, player, currentKey])

  const updatePlayer = () => {
    // currentKey is at default value here...
    const direction = currentKey === KEYS.left ? -1 : currentKey === KEYS.right ? 1 : null;


    if(direction !== null) move(direction)

  }

  const animate = (time) => {

    var now = window.performance.now();
    var dt = now - lastTime;

    if(dt > 100) {
      lastTime = now;
      updatePlayer();
    };

    requestAnimationFrame(animate);


  }



  return (
    <canvas
      className={classes.canvas}
      ref={canvasRef}
      width={GAME.shape.w}
      height={GAME.shape.h}

    />
  )


}

export default SpaceInvader;

Я пытаюсь получить доступ к currentKey"в функции потока, но она всегда возвращает" ноль "(значение состояния по умолчанию),

Я обнаружил, что вам нужно связать контекст с функцией animate, но я не знаю, как это сделать. это с функциональным компонентом (с компонентом класса я бы сделал .bind (this))

Я довольно новичок в HTML5 canvas, поэтому я не смогу увидеть проблему здесь.

Все советы приветствуются,

Заранее спасибо!

1 Ответ

0 голосов
/ 02 ноября 2019

Наконец, после многих испытаний я сделал это:

React.useEffect(() => {
    console.log("use Effect");
    if(!cctx) return
    requestRef.current = requestAnimationFrame(animate);
    return () => cancelAnimationFrame(requestRef.current);
  })

, и это работает ...

Если у кого-нибудь есть более чистое решение

РЕДАКТИРОВАТЬ 1: Послееще какое-то тестирование, это может быть не жизнеспособным решением, оно вызывает слишком много повторного рендеринга

...