在 Javascript 中调整重复图像的大小

Resize Repeated Image in Javascript

我的代码使用二维数组来知道在网格布局上显示什么图像。使用的图像是 repeated.The 图像是在 javascript 文件中用 var ground = new Image() 创建的。我正在尝试调整图像的大小,但我无法弄清楚。我尝试过使用各种组合,例如 ground.value.height="10px"ground.height="10px"ground.value.height="10"ground.value.height=10

更具体地说,我想要的是图像的大小 canvas 除以二维数组的相关维度。即网格中的图块大小可以是 10px x 10px,我希望图像大小相同。如果它按照我认为的方式运行,那么当图像重复时,每个图块都应该重复显示的图像。

var gameContext = null;
var gameMap = [
  [0, 0, 0, 0, 0, 0, 1, 0, 0, 0],
  [0, 1, 1, 1, 0, 1, 1, 1, 1, 0],
  [0, 1, 0, 0, 0, 1, 0, 0, 0, 0],
  [0, 1, 1, 1, 1, 1, 1, 1, 1, 0],
  [0, 1, 0, 1, 0, 0, 0, 1, 1, 0],
  [0, 1, 0, 1, 0, 1, 0, 0, 1, 0],
  [0, 1, 1, 1, 1, 1, 1, 1, 1, 0],
  [0, 1, 0, 0, 0, 0, 0, 1, 0, 0],
  [0, 1, 1, 1, 0, 1, 1, 1, 1, 0],
  [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
];

var mapW = 10,
  mapH = 10;
var boxEdge = window.innerWidth / mapW / 2;
var tileW = boxEdge,
  tileH = boxEdge;
var currentSecond = 0,
  frameCount = 0,
  framesLastSecond = 0;
var ground = new Image();
ground.src = "https://images-wixmp-ed30a86b8c4ca887773594c2.wixmp.com/f/65614919-0734-4dc8-9460-7034fd979346/dbg8qqd-0fb0aced-d05c-4df6-a7c6-b8e04c184ac5.png?token=eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJzdWIiOiJ1cm46YXBwOjdlMGQxODg5ODIyNjQzNzNhNWYwZDQxNWVhMGQyNmUwIiwiaXNzIjoidXJuOmFwcDo3ZTBkMTg4OTgyMjY0MzczYTVmMGQ0MTVlYTBkMjZlMCIsIm9iaiI6W1t7InBhdGgiOiJcL2ZcLzY1NjE0OTE5LTA3MzQtNGRjOC05NDYwLTcwMzRmZDk3OTM0NlwvZGJnOHFxZC0wZmIwYWNlZC1kMDVjLTRkZjYtYTdjNi1iOGUwNGMxODRhYzUucG5nIn1dXSwiYXVkIjpbInVybjpzZXJ2aWNlOmZpbGUuZG93bmxvYWQiXX0.9IQbxC3HC3uuJLf8V9Ridq005b2_-4zFg6Cb9rJ2tbw";
var wall = new Image();
wall.src = "";
window.onload = function() {
  gameCanvas = document.getElementById("game");
  gameContext = gameCanvas.getContext("2d");
  gameCanvas.width = window.innerWidth / 2;
  gameCanvas.height = window.innerWidth / 2;
  /*gameContext = document.getElementById('game').getContext("2d");
  gameContext.font = "bold 10pt sans-serif";*/
  requestAnimationFrame(drawGame);
};

function drawGame() {
  var groundPattern = gameContext.createPattern(ground, "repeat");
  var wallPattern = gameContext.createPattern(wall, "repeat");
  if (gameContext == null) {
    return;
  }

  var sec = Math.floor(Date.now() / 1000);
  if (sec != currentSecond) {
    currentSecond = sec;
    framesLastSecond = frameCount;
    frameCount = 1;
  } else {
    frameCount++;
  }

  for (var y = 0; y < mapH; ++y) {
    for (var x = 0; x < mapW; ++x) {
      switch (gameMap[y][x]) {
        case 0:
          gameContext.fillStyle = wallPattern;
          break;
        default:
          gameContext.fillStyle = groundPattern;
          //gameContext.fillStyle = "#5aa457";
      }

      gameContext.fillRect(x * tileW, y * tileH, tileW, tileH);
    }
  }

  //gameContext.fillStyle = "#ff0000";
  //gameContext.fillText("FPS: " + framesLastSecond, 10, 20);

  requestAnimationFrame(drawGame);
}
<!DOCTYPE html>
<html>

<head>
  <link rel="stylesheet" href="style.css">
</head>

<body style="background-color:powderblue;">
  <script src="script.js"></script>
  <canvas id="game"></canvas>
</body>

</html>

您可以将图块绘制为缩放图像而不是图案:

ctx.drawImage(
  imgAsset.img,
  col * tile.width,
  row * tile.height,
  tile.width,
  tile.height
);

此示例使用 CanvasRenderingContext2D.drawImage() 的第二个构造函数:

void ctx.drawImage(image, dx, dy, dWidth, dHeight);

在下面的示例中,我添加了资产查找地图以提高效率。您可以添加额外的图像资源并相应地更新您的瓷砖地图。该代码仍然有效。

const assets = {
  images: {
    wall: {
      id: 0,
      url: 'https://cdn.discordapp.com/attachments/808770424535777302/850378944653164604/wall.png'
    },
    ground: {
      id: 1,
      url: 'https://cdn.discordapp.com/attachments/808770424535777302/850378948168384562/ground.png'
    },
    water: {
      id: 2,
      url: 'https://cdn.discordapp.com/attachments/808770424535777302/850398765726302298/water.png'
    }
  }
};
const imgKeyLookup = Object.entries(assets.images)
  .reduce((acc, [key, { id }]) => acc.set(id, key), new Map);
const getImageAsset = id => assets.images[imgKeyLookup.get(id)];

let currentSecond = 0, frameCount = 0, framesLastSecond = 0;

const tileMap = [
  [0, 0, 0, 0, 0, 0, 1, 0, 0, 0],
  [0, 1, 1, 1, 0, 1, 1, 1, 1, 0],
  [0, 1, 0, 0, 0, 1, 0, 0, 0, 0],
  [0, 1, 1, 1, 1, 1, 1, 1, 1, 0],
  [0, 1, 0, 1, 2, 2, 2, 1, 1, 0],
  [0, 1, 0, 1, 2, 2, 0, 0, 1, 0],
  [0, 1, 1, 1, 1, 1, 1, 1, 1, 0],
  [0, 1, 0, 0, 0, 0, 0, 1, 0, 0],
  [0, 1, 1, 1, 0, 1, 1, 1, 1, 0],
  [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
];

const ctx = document.querySelector('#game').getContext('2d');
const grid = { rows: tileMap.length, cols: tileMap[0].length };
const boxEdge = window.innerWidth / grid.cols / 2;
const tile = { width: boxEdge, height: boxEdge };

const loadImage = (url) => new Promise((resolve, reject) => {
  const img = new Image();
  img.addEventListener('load', () => resolve(img));
  img.addEventListener('error', (err) => reject(err));
  img.src = url;
});

const main = () => {
  Object.assign(ctx.canvas, {
    width: window.innerWidth / 2,
    height: window.innerWidth / 2
  });
  requestAnimationFrame(drawGame);
};

function drawGame() {
  const sec = Math.floor(Date.now() / 1000);
  if (sec != currentSecond) {
    currentSecond = sec;
    framesLastSecond = frameCount;
    frameCount = 1;
  } else {
    frameCount++;
  }

  for (let row = 0; row < grid.rows; row++) {
    for (let col = 0; col < grid.cols; col++) {
      const imgAsset = getImageAsset(tileMap[row][col]);
      if (imgAsset) {
        ctx.drawImage(imgAsset.img,
          col * tile.width, row * tile.height,
          tile.width, tile.height);
      }
    }
  }
  
  requestAnimationFrame(drawGame);
};

const imageAssets = Object.values(assets.images);
Promise.all(imageAssets.map(({ url }) => url).map(loadImage))
  .then(imgArr => imgArr.forEach((img, index) => {
    Object.assign(imageAssets[index], { img });
  }))
  .then(main);
html, body {
  width: 100%;
  height: 100%;
  margin: 0;
  padding: 0;
}

body {
  display: flex;
  background-color: powderblue;
  align-items: center;
  justify-content: center;
}
<canvas id="game"></canvas>


设置 Image 构造函数的 widthheight 不会调整您的图块图像的大小(如下所示)。

const assets = {
  images: {
    wall: {
      id: 0,
      url: 'https://cdn.discordapp.com/attachments/808770424535777302/850378944653164604/wall.png'
    },
    ground: {
      id: 1,
      url: 'https://cdn.discordapp.com/attachments/808770424535777302/850378948168384562/ground.png'
    }
  }
};
const imgKeyLookup = Object.entries(assets.images)
  .reduce((acc, [key, { id }]) => acc.set(id, key), new Map);
const getImageAsset = id => assets.images[imgKeyLookup.get(id)];

let currentSecond = 0, frameCount = 0, framesLastSecond = 0;

const tileMap = [
  [0, 0, 0, 0, 0, 0, 1, 0, 0, 0],
  [0, 1, 1, 1, 0, 1, 1, 1, 1, 0],
  [0, 1, 0, 0, 0, 1, 0, 0, 0, 0],
  [0, 1, 1, 1, 1, 1, 1, 1, 1, 0],
  [0, 1, 0, 1, 0, 0, 0, 1, 1, 0],
  [0, 1, 0, 1, 0, 1, 0, 0, 1, 0],
  [0, 1, 1, 1, 1, 1, 1, 1, 1, 0],
  [0, 1, 0, 0, 0, 0, 0, 1, 0, 0],
  [0, 1, 1, 1, 0, 1, 1, 1, 1, 0],
  [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
];

const ctx = document.querySelector('#game').getContext('2d');
const grid = { rows: tileMap.length, cols: tileMap[0].length };
const boxEdge = window.innerWidth / grid.cols / 2;
const tile = { width: boxEdge, height: boxEdge };

const loadImage = (url) => new Promise((resolve, reject) => {
  const img = new Image(tile.width, tile.height);
  img.addEventListener('load', () => resolve(img));
  img.addEventListener('error', (err) => reject(err));
  img.src = url;
});

const main = () => {
  Object.entries(assets.images).forEach(([key, value]) =>
    Object.assign(value, {
      pattern: ctx.createPattern(value.img, 'repeat')
    }));
  Object.assign(ctx.canvas, {
    width: window.innerWidth / 2,
    height: window.innerWidth / 2
  });
  requestAnimationFrame(drawGame);
};

function drawGame() {
  const sec = Math.floor(Date.now() / 1000);
  if (sec != currentSecond) {
    currentSecond = sec;
    framesLastSecond = frameCount;
    frameCount = 1;
  } else {
    frameCount++;
  }

  for (let row = 0; row < grid.rows; row++) {
    for (let col = 0; col < grid.cols; col++) {
      const imgAsset = getImageAsset(tileMap[row][col]);
      if (imgAsset) {
        ctx.fillStyle = imgAsset.pattern;
        ctx.fillRect(col * tile.width, row * tile.height, tile.width, tile.height);
      }
    }
  }
  
  requestAnimationFrame(drawGame);
};

const imageAssets = Object.values(assets.images);
Promise.all(imageAssets.map(({ url }) => url).map(loadImage))
  .then(imgArr => imgArr.forEach((img, index) => {
    Object.assign(imageAssets[index], { img });
  }))
  .then(main);
html, body {
  width: 100%;
  height: 100%;
  margin: 0;
  padding: 0;
}

body {
  display: flex;
  background-color: powderblue;
  align-items: center;
  justify-content: center;
}
<canvas id="game"></canvas>

图像的大小应该传递给构造函数。

如果你想要 10px10px 图片,你需要写:

var image = new Image(10, 10)

Based on the https://developer.mozilla.org/en-US/docs/Web/API/HTMLImageElement/Image