如何在视口上调整画布宽度

时间:2018-12-25 04:16:50

标签: javascript html css canvas html5-canvas

我有一个画布演示,可以产生一些圆圈。我似乎无法弄清楚如何在使用{{1来调整大小的窗口上设置canvas.widthcanvas.style.width (这里有什么区别?) }}。

代码工作正常,但在较小的视口上却奇怪地重新渲染。我尝试在JavaScript文件的底部添加此代码段,但它破坏了动画效果。

window.innerWidth

enter image description here

  // ADDING THESE LINES PREVENTS ANIMATION FROM RUNNING
  if (window.innerWidth < 1000) {
    canvas.width = window.innerWidth;
   } else {
     canvas.width = 1000;
   }
var canvas = document.querySelector("canvas");

canvas.width = 1000;
canvas.height = 100;
var c = canvas.getContext("2d");

// Constructor Function (object blueprint)
function Circle(x, y, dx, dy, radius, counter) {
  this.x = x;
  this.y = y;
  this.dx = dx;
  this.dy = dy;
  this.radius = radius;
  this.counter = counter;

  this.draw = function() {
    c.beginPath();
    c.arc(this.x, this.y, this.radius, 0, Math.PI * 2, false);
    c.strokeStyle = "white";
    c.stroke();
    c.fillStyle = "white";
    c.fill();
  };

  this.update = function() {
    if (this.y + this.radius > canvas.height) {
      this.y = 0;
    }
    this.x += this.dx;
    this.y += this.dy;

    this.draw();
  };
}

// Initialize array to store snow objects
var circleArray = [];

// Initialize objects with constructor
for (var i = 0; i < 50; i++) {
  var radius = 1 + Math.random() * 5;
  var x = Math.random() * canvas.width;
  var y = 0 - Math.random() * 50; // start at top, render some circles off screen
  var dx = (Math.random() - 0.5) * 2;
  var dy = 0.5 + Math.random() * 0.5; // use gravity
  circleArray.push(new Circle(x, y, dx, dy, radius, 0));
}

function animate() {
  requestAnimationFrame(animate); // recurisvely run
  c.clearRect(0, 0, innerWidth, innerHeight); // erases previously drawn content

  for (var i = 0; i < circleArray.length; i++) {
    circleArray[i].update();
  }
  // ADDING THESE LINES PREVENTS ANIMATION FROM RUNNING
  //if (window.innerWidth < 1000) {
  //  canvas.width = window.innerWidth;
  // } else {
    // canvas.width = 1000;
  // }
}
animate();
body {
  background-color: grey;
  display: flex;
  justify-content: center;
}
.wrapper {
  position: relative;
  width: 1000px;
  height: 110px;
  min-height: 110px;
  margin-top: 50vh;
}
.wrapper > * {
  width: 1000px;
  position: absolute;
}
canvas {
  position: absolute;
}
img {
  width: 100%;
  height: 110px;
}

1 个答案:

答案 0 :(得分:3)

canvas.widthcanvas.style.width之间的区别在于canvas.width指定画布的实际大小(以像素为单位),而canvas.style.width 拉伸或压缩画布达到您指定的宽度。这会将画布宽度设置为300px:

canvas.width = 300;

这还将将画布宽度设置为300px,但将其拉伸到600px:

canvas.width = 300;
canvas.style.width = "600px";

使用canvas.width是更好的做法。


要运行动画,必须首先固定画布宽度。将调整画布大小的代码放在animate()函数的开头即可解决此问题:

var canvas = document.querySelector("canvas");

canvas.width = 1000;
canvas.height = 100;
var c = canvas.getContext("2d");

// Constructor Function (object blueprint)
function Circle(x, y, dx, dy, radius, counter) {
  this.x = x;
  this.y = y;
  this.dx = dx;
  this.dy = dy;
  this.radius = radius;
  this.counter = counter;

  this.draw = function() {
    c.beginPath();
    c.arc(this.x, this.y, this.radius, 0, Math.PI * 2, false);
    c.strokeStyle = "white";
    c.stroke();
    c.fillStyle = "white";
    c.fill();
  };

  this.update = function() {
    if (this.y + this.radius > canvas.height) {
      this.y = 0;
    }
    this.x += this.dx;
    this.y += this.dy;

    this.draw();
  };
}

// Initialize array to store snow objects
var circleArray = [];

// Initialize objects with constructor
for (var i = 0; i < 50; i++) {
  var radius = 1 + Math.random() * 5;
  var x = Math.random() * canvas.width;
  var y = 0 - Math.random() * 50; // start at top, render some circles off screen
  var dx = (Math.random() - 0.5) * 2;
  var dy = 0.5 + Math.random() * 0.5; // use gravity
  circleArray.push(new Circle(x, y, dx, dy, radius, 0));
}

function animate() {
  if (window.innerWidth < 1000) {
    canvas.width = window.innerWidth;
  } else {
    canvas.width = 1000;
  }
  
  requestAnimationFrame(animate); // recurisvely run
  c.clearRect(0, 0, innerWidth, innerHeight); // erases previously drawn content

  for (var i = 0; i < circleArray.length; i++) {
    circleArray[i].update();
  }
}
animate();
body {
  background-color: grey;
  display: flex;
  justify-content: center;
}

.wrapper {
  position: relative;
  width: 1000px;
  height: 110px;
  min-height: 110px;
  margin-top: 50vh;
}

.wrapper>* {
  width: 1000px;
  position: absolute;
}

canvas {
  position: absolute;
}

img {
  width: 100%;
  height: 110px;
}
<div class="wrapper">
  <img class="stars" src="https://s3-us-west-2.amazonaws.com/s.cdpn.io/867725/stars.png
                          " alt="" />
  <img class="tress" src="https://s3-us-west-2.amazonaws.com/s.cdpn.io/867725/trees.png
                          " alt="" />
  <img class="clouds" src="https://s3-us-west-2.amazonaws.com/s.cdpn.io/867725/clouds.png" alt="" />
  <canvas></canvas>
</div>