在 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 构造函数的 width
和 height
不会调整您的图块图像的大小(如下所示)。
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>
图像的大小应该传递给构造函数。
如果你想要 10px 的 10px 图片,你需要写:
var image = new Image(10, 10)
Based on the https://developer.mozilla.org/en-US/docs/Web/API/HTMLImageElement/Image
我的代码使用二维数组来知道在网格布局上显示什么图像。使用的图像是 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 构造函数的 width
和 height
不会调整您的图块图像的大小(如下所示)。
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>
图像的大小应该传递给构造函数。
如果你想要 10px 的 10px 图片,你需要写:
var image = new Image(10, 10)
Based on the https://developer.mozilla.org/en-US/docs/Web/API/HTMLImageElement/Image