Холст лабиринт игра плавная анимация - PullRequest
0 голосов
/ 12 февраля 2019

Я создаю игру-лабиринт на основе этого учебника.Я успешно получил прямоугольник игрока, чтобы продолжать двигаться, пока вы удерживаете клавиши со стрелками.Когда вы впервые запускаете игру, анимация действительно приятная и быстрая, но кажется, что после игры в течение нескольких секунд анимация становится все медленнее и медленнее .Может кто-нибудь помочь мне выяснить, почему это происходит?

Я создал фрагмент кода, но, к сожалению, он не работает должным образом из-за ошибки перекрестного источника, вызванной использованием изображения лабиринта.

var canvas;
var ctx;
var dx = 2;
var dy = 2;
var WIDTH = 482;
var HEIGHT = 482;

//movement
var x = 200,
    y = 5,
    staticX = 200,//this should be the same as x (used for resetting the game)
    staticY = 5,//this should be the same as y (used for resetting the game)
    keys = [];

var img = new Image();
var collision = 0;
var showingWinScreen = false;
var playerSize = 15;

var startTime = null,
  lastTime = null,
  endTime,  // for scale
  isRunning = false,
  FPS = 1000/60; // ideal frame rate


function rect(x,y,w,h) {
  ctx.beginPath();
  ctx.rect(x,y,w,h);
  ctx.closePath();
  ctx.fill();
}

function clear() {
  ctx.clearRect(0, 0, WIDTH, HEIGHT);
}

function drawMaze() {
  ctx.drawImage(img, 0, 0);
}

function drawPlayer() {
  doKeyDown();

  ctx.fillStyle = "purple";
  rect(x, y, 15,15);
}

function draw() {
  clear();

  if(showingWinScreen) {
    isRunning = false;
    drawRect(0,0,canvas.width, canvas.height,"black");
    ctx.fillStyle = "white";
    ctx.font = "20px Arial";
    ctx.fillText("You Won! Click to play again", 70,canvas.height/2);
    ctx.fillText("Time Elapsed: " + endTime, 70, canvas.height*0.6);
    return;
  }
  isRunning = true;

  drawMaze();
  drawPlayer();
  requestAnimationFrame(loop);
}

function drawRect(leftX, topY, width, height, drawColor) {
  ctx.fillStyle = drawColor;
  ctx.fillRect(leftX,topY,width, height);
}

//timer
function loop(timeStamp) {

  if (!startTime) {
    startTime = timeStamp;
  }

  var timeDiff = lastTime ? timeStamp - lastTime : FPS,
      timeElapsed = timeStamp - startTime,
      timeScale = timeDiff / FPS; // adjust variations in frame rates

  lastTime = timeStamp;

  var totalTime = timeElapsed*0.001;
  var minutes = Math.floor(totalTime / 60);
  var seconds = totalTime % 60;

  drawRect(WIDTH,10,70, 30,"black");
  ctx.fillStyle = "white";
  ctx.font = "14px Arial";
  ctx.fillText(minutes + ":" + (seconds).toFixed(2), WIDTH*1.04, 30);

  endTime = minutes + ":" + (seconds).toFixed(0);

  if (isRunning) requestAnimationFrame(loop);
}

function init() {

  canvas = document.getElementById("canvas");
  ctx = canvas.getContext("2d");
  img.src = "https://html5.litten.com/images/maze.gif";
  var framesPerSecond = 60;
  return setInterval( function() {
    draw();
  }, 1000/framesPerSecond);
}

function checkcollision() {
  var imgd = ctx.getImageData(x, y, playerSize, playerSize);
  var pix = imgd.data;
  for (var i = 0; n = pix.length, i < n; i += 4) {
    if (pix[i] == 0) {
      collision = 1;
    }
  }
}

function checkWin() {
  var imageData = ctx.getImageData(x, y, playerSize, playerSize);
  var r, g, b, a;
  for (var i = 0; i+3 < imageData.data.length; i += 4) {
    r = imageData.data[i];
    g = imageData.data[i+1];
    b = imageData.data[i+2];
    a = imageData.data[i+3];

    //if red
    if ( r === 255 && b === 0 ) {
      console.log(' R: ' + r + '<br>G: ' + g + '<br>B: ' + b);
      isRunning = false;
      showingWinScreen = true;
    }
  }
}



function handleMouseClick(event) {
  if (showingWinScreen) {
    showingWinScreen = false;
    x = staticX;
    y = staticY;
    draw();
  }
}

//arrow keys
function doKeyDown(){
  //left
  if (keys[37]) {
    if (x - dx > 0){
      x -= dx;
      checkcollision();
      checkWin();
      if (collision == 1){
        x += dx;
        collision = 0;
      }
    }
  }
  //right
  if (keys[39]) {
    if (x + dx < WIDTH){
      x += dx;
      checkcollision();
      checkWin();
      if (collision == 1){
        x -= dx;
        collision = 0;
      }
    }
  }
  //down
  if (keys[40]) {
    if (y + dy < HEIGHT ){
      y += dy;
      checkcollision();
      checkWin();
      if (collision == 1){
        y -= dy;
        collision = 0;
      }
    }
  }
  //up
  if (keys[38]) {
    if (y - dy > 0){
      y -= dy;
      checkcollision();
      checkWin();
      if (collision == 1){
        y += dy;
        collision = 0;
      }
    }
  }
}

init();

window.addEventListener("keydown", function (e) {
  keys[e.keyCode] = true;
});
window.addEventListener("keyup", function (e) {
  keys[e.keyCode] = false;
});

canvas.addEventListener("mousedown", handleMouseClick);

(function () {
  var requestAnimationFrame = window.requestAnimationFrame || window.mozRequestAnimationFrame || window.webkitRequestAnimationFrame || window.msRequestAnimationFrame;
  window.requestAnimationFrame = requestAnimationFrame;
})()
<canvas id="canvas" width="582" height="582">
      This text is displayed if your browser does not support HTML5 Canvas.
      </canvas>

1 Ответ

0 голосов
/ 13 февраля 2019

Самая большая проблема здесь?
Ваш таймер.

Давайте удалим все, что связано с холстом, и просто попробуем записать, сколько раз за кадр вы рисуете этот маленький счетчик времени:

// If everything was ok, frame_count should never be higher than 1
var frame_count = 0;
var total_count = 0;

function frame_loop() {
  frame_log.textContent = frame_count;
  total_log.textContent = total_count;
  // reset our frame counter
  frame_count = 0;
  // do it again next loop
  requestAnimationFrame(frame_loop);
}
frame_loop();

var startTime = null,
  lastTime = null,
  endTime,  // for scale
  isRunning = false,
  FPS = 1000/60; // ideal frame rate

function draw() {
  isRunning = true;
  requestAnimationFrame(loop);
}

//timer
function loop(timeStamp) {

  frame_count ++;
  total_count ++;
  
  if (!startTime) {
    startTime = timeStamp;
  }

  var timeDiff = lastTime ? timeStamp - lastTime : FPS,
      timeElapsed = timeStamp - startTime,
      timeScale = timeDiff / FPS; // adjust variations in frame rates

  lastTime = timeStamp;

  var totalTime = timeElapsed*0.001;
  var minutes = Math.floor(totalTime / 60);
  var seconds = totalTime % 60;
  endTime = minutes + ":" + (seconds).toFixed(0);
  if (isRunning) requestAnimationFrame(loop);
}

function init() {
  var framesPerSecond = 60;
  return setInterval( function() {
    draw();
  }, 1000/framesPerSecond);
}


init();
<p>number of times loop() has been called <b>during last frame</b>: <span id="frame_log"></span></p>

<p>number of times loop() has been called <b>in total</b>: <span id="total_log"></span></p>

Вы видите проблему?
Через несколько секунд вы отрисовываете этот текст тысячи раз за кадр .
Его нужно рисовать только один раз за кадр.

Это потому, что вы смешали setInterval и requestAnimationFrame.

Как правило, никогда не делайте этого.
requestAnimationFrame должен вызываться только из обратного вызова или из функции инициализации.

setInterval не должен использоваться для вызова чего-либо, что должно быть нарисовано с высокой частотой.Это работа requestAnimationFrame.

Так что избавьтесь от этого setInterval и используйте один цикл requestAnimationFrame:

// If everything was ok, frame_count should never be higher than 1
var frame_count = 0;
var total_count = 0;

function frame_loop() {
  frame_log.textContent = frame_count;
  total_log.textContent = total_count;
  // reset our frame counter
  frame_count = 0;
  // do it again next loop
  requestAnimationFrame(frame_loop);
}
frame_loop();

var startTime = null,
  lastTime = null,
  endTime,  // for scale
  isRunning = false,
  FPS = 1000/60; // ideal frame rate

function draw() {
  isRunning = true;
  // remove this one, loop will call itself
//  requestAnimationFrame(loop);
}

//timer
function loop(timeStamp) {

  frame_count ++;
  total_count ++;
  
  if (!startTime) {
    startTime = timeStamp;
  }

  var timeDiff = lastTime ? timeStamp - lastTime : FPS,
      timeElapsed = timeStamp - startTime,
      timeScale = timeDiff / FPS; // adjust variations in frame rates

  lastTime = timeStamp;

  var totalTime = timeElapsed*0.001;
  var minutes = Math.floor(totalTime / 60);
  var seconds = totalTime % 60;
  endTime = minutes + ":" + (seconds).toFixed(0);

  if (isRunning) requestAnimationFrame(loop);
}

function init() {
  isRunning  = true;
  // from init only we can start the loop
  loop();
}


init();
<p>number of times loop() has been called <b>during last frame</b>: <span id="frame_log"></span></p>

<p>number of times loop() has been called <b>in total</b>: <span id="total_log"></span></p>

Теперь, когда мы исправили эту огромную проблему, мы можем проверить немного больше того, что вы делаете ...

Делатьне смешивать игровую логику с рендерингом.

Базовая настройка - это один основной цикл, который будет вызывать все подфункции и сам.
Эти подфункции в основном

  • 1 обновление сцены
  • 2 рендеринг сцены

Внутри части обновления вы обновите положение игрока, проверите столкновения и т. Д.

Тогда для рендеринга нужно будет использовать обновленные значения.

(function(imgurl) {

  function mainLoop(t) {
    update(t);
    render();
    if (isRunning) {
      requestAnimationFrame(mainLoop);
    }
  }

  function update(t) {
    updateTimer(t);
    updatePlayer(t);
  }

  function render() {
    clear();

    if (showingWinScreen) {
      showWinScreen();
      return;
    }

    renderTimer();
    renderMaze();
    renderPlayer();
  }

  function init() {
    canvas = document.getElementById("canvas");
    ctx = canvas.getContext("2d");

    img.onload = mainLoop;
    img.src = imgurl;

    canvas.addEventListener("mousedown", handleMouseClick);

    isRunning = true;
  }

  function showWinScreen() {
    isRunning = false;
    drawRect(0, 0, canvas.width, canvas.height, "black");
    ctx.fillStyle = "white";
    ctx.font = "20px Arial";
    ctx.fillText("You Won! Click to play again", 70, canvas.height / 2);
    ctx.fillText("Time Elapsed: " + endTime, 70, canvas.height * 0.6);
  }

  var canvas;
  var ctx;
  var dx = 2;
  var dy = 2;
  var WIDTH = 482;
  var HEIGHT = 482;

  //movement
  var x = 200,
    y = 5,
    staticX = 200,
    staticY = 5,
    keys = [];

  var img = new Image();
  var collision = 0;
  var showingWinScreen = false;
  var playerSize = 15;


  //timer
  var startTime = null,
    lastTime = null,
    endTime, // for scale
    isRunning = false,
    timer = '';

  function updateTimer(timeStamp) {

    if (!startTime) {
      startTime = timeStamp;
    }

    var timeElapsed = timeStamp - startTime;

    lastTime = timeStamp;

    var totalTime = timeElapsed * 0.001;
    var minutes = Math.floor(totalTime / 60);
    var seconds = totalTime % 60;

    timer = minutes + ":" + (seconds).toFixed(2);
    endTime = minutes + ":" + (seconds).toFixed(0);

  }

  function renderTimer() {
    drawRect(WIDTH, 10, 70, 30, "black");
    ctx.fillStyle = "white";
    ctx.font = "14px Arial";
    ctx.fillText(timer, WIDTH * 1.04, 30);
  }

  // merge both collision and win in a single check
  function checkPosition() {
    collision = 0;
    showingWinScreen = false;
    var imgd = ctx.getImageData(x, y, playerSize, playerSize);
    var pix = imgd.data;
    var r, g, b, a;
    for (var i = 0, n = pix.length; i < n; i += 4) {
      r = pix[i];
      g = pix[i + 1];
      b = pix[i + 2];
      a = pix[i + 3];
      if (r == 0) {
        collision = 1;
      }
      //if red
      if (r === 255 && b === 0) {
        console.log(' R: ' + r + '<br>G: ' + g + '<br>B: ' + b);
        isRunning = false;
        showingWinScreen = true;
      }
    }
  }

  function updatePlayer() {

    var direction_x = 0;
    var direction_y = 0;

    // get the directions
    if (keys[37]) { //left
      direction_x = -dx;
    }
    if (keys[39]) { //right
      direction_x += dx;
    }
    if (keys[40]) { //bottom
      direction_y += dy;
    }
    if (keys[38]) { //top
      direction_y -= dy;
    }

    var updated = false;

    // update the position
    if (x + direction_x > 0 && x + direction_x < WIDTH) {
      x += direction_x;
      updated = true;
    }
    if (y + direction_y > 0 && y + direction_y < HEIGHT) {
      y += direction_y;
      updated = true;
    }

    // check for collision/win
    if (updated) {
      checkPosition();
    }

    // undo if needed
    if (collision === 1) {
      x -= direction_x;
      y -= direction_y;
    }
  }

  function renderPlayer() {
    drawRect(x, y, playerSize, playerSize, "purple");
  }

  function renderMaze() {
    ctx.drawImage(img, 0, 0);
  }

  function clear() {
    ctx.clearRect(0, 0, WIDTH, HEIGHT);
  }

  function drawRect(leftX, topY, width, height, drawColor) {
    ctx.fillStyle = drawColor;
    ctx.fillRect(leftX, topY, width, height);
  }

  function handleMouseClick(event) {
    if (showingWinScreen) {
      showingWinScreen = false;
      x = staticX;
      y = staticY;
      mainLoop(); // restart
    }
  }

  window.addEventListener("keydown", function(e) {
    e.preventDefault();
    keys[e.keyCode] = true;
  });
  window.addEventListener("keyup", function(e) {
    keys[e.keyCode] = false;
  });

  init();

})('');
<canvas id="canvas" width="582" height="582"></canvas>

Есть еще много вещей, которые можно улучшить, например, вам лучше иметь лабиринт в формате JSON и смотреть только на x и y значения для столкновения и выигрыша вместо проверки нарисованного пикселя, но это было бы слишком много для этого небольшого ответа.

...