如何创建带有图像的画布圆圈?

问题描述 投票:0回答:2

所以,我按照教程制作了一个游戏。显然,这不是学习编码的最佳方式,所以我开始修改它。目前游戏中有慢慢向玩家移动的敌人,但这些敌人只是彩色圆圈,没有其他东西。我想添加一张可以放在敌人身上的图片,但我不知道该怎么做。以下是您可能想知道的一些代码:

敌人类别(每帧调用更新):

class Enemy {
    constructor(x, y, radius, color, velocity) {
        this.x = x;
        this.y = y;
        this.radius = radius;
        this.color = color;
        this.velocity = velocity;
    }

    draw() {
        ctx.beginPath();
        ctx.arc(this.x, this.y, this.radius, 0, Math.PI * 2, true);
        ctx.fillStyle = this.color;
        ctx.fill();
    }

    update() {
        this.draw();
        this.x = this.x + this.velocity.x;
        this.y = this.y + this.velocity.y;
    }
}

创建敌人功能:

function spawnEnemies() {
    setInterval(() => {
        const radius = Math.random() * (30 - 4) + 4;
        let x;
        let y;
        if (Math.random() < 0.5) {
            x = Math.random() < 0.5 ? 0 - radius : canvas.width + radius;
            y = Math.random() * canvas.height;
        } else {
            y = Math.random() < 0.5 ? 0 - radius : canvas.height + radius;
            x = Math.random() * canvas.width;
        }
        const color = `hsl(${Math.random() * 360}, 50%, 50%)`;
        const angle = Math.atan2(canvas.height / 2 - y, canvas.width / 2 - x);
    const velocity = {
        x: Math.cos(angle),
        y: Math.sin(angle)
    }
        enemies.push(new Enemy(x, y, radius, color, velocity));
    }, 1000)
}

此代码在 animate 函数中运行:

enemies.forEach((enemy, index) => {
        enemy.update();
        const dist = Math.hypot(player.x - enemy.x, player.y - enemy.y);
        
        if (isHacking) {
            if (dist - enemy.radius - player.radius < 11) {
                setTimeout(() => {
                for (let i = 0; i < enemy.radius * 2; i++) {
                    particles.push(new Particle(enemy.x, enemy.y, Math.random() * 2, enemy.color, {
                        x: (Math.random() - 0.5) * (Math.random() * 8),
                        y: (Math.random() - 0.5) * (Math.random() * 8)}
                    ));
                }}, 0)
                score += 25;
                scoreEl.innerHTML = score;
                setTimeout(() => {
                    enemies.splice(index, 1);
                    projectiles.splice(proIndex, 1);
                }, 0)
            }         
        } else if (dist - enemy.radius - player.radius < 1) {
            cancelAnimationFrame(animationId);
            modal.style.display = 'flex';
            modalScore.innerHTML = score;
        }

另外,这是我第一次在堆栈溢出上发帖,所以如果有什么我应该做而没有做的事情,或者反之亦然,请告诉我!

javascript html5-canvas
2个回答
1
投票

有很多方法可以做你想做的事。带图像的圆圈。

从你的问题和评论中我可以猜测你想要一个显示加载图像的动态圆圈。

使用图像创建图案。

const imgPat = ctx.createPattern(image, "no-repeat");

使用图像大小来计算在不超出图像范围的情况下可以拥有的最大半径。

const minRadius = Math.min(image.width, image.height) / 2;

要以任意半径绘制圆,您需要使用 2D context

setTransform
函数。下面的函数可以做到这一点

function drawImageCircle(imgPat, minRadius, x, y, radius) {

     // get scale of circle image
     const scale = radius / minRadius;

     // transform to put origin at top left of bounding rectangle scaling to fit image pattern
     ctx.setTransform(scale, 0, 0, scale, x - radius, y - radius);
     ctx.fillStyle = imgPat;
     ctx.beginPath();
     ctx.arc(minRadius, minRadius, minRadius, 0, Math.PI * 2);  
     ctx.fill();

     // reset the default transform 
     ctx.setTransform(1, 0, 0, 1, 0, 0);

}

演示

演示加载图像。然后获取它的大小,从中创建一个图案,并对其进行动画处理,更改半径并在其周围放置 4 像素的轮廓。

const image = new Image;
image.src = "https://i.stack.imgur.com/C7qq2.png?s=256&g=1";
image.addEventListener("load", imageReady);
const ctx = canvas.getContext("2d");
var w = canvas.width, h = canvas.height, cw = w / 2, ch = h / 2;
var imgPat, minRadius;

function imageReady() {
    imgPat = ctx.createPattern(image, "no-repeat");
    minRadius = Math.min(image.width, image.height) / 2;
    requestAnimationFrame(renderLoop);
}

function drawImageCircle(imgPat, minRadius, x, y, radius) {
     const scale = radius / minRadius;
     ctx.setTransform(scale, 0, 0, scale, x - radius, y - radius);
     ctx.strokeStyle = "#F00";
     ctx.lineWidth = 8 / scale;
     ctx.fillStyle = imgPat;
     ctx.beginPath();
     ctx.arc(minRadius, minRadius, minRadius, 0, Math.PI * 2); 
     ctx.stroke();
     ctx.fill();
     ctx.setTransform(1, 0, 0, 1, 0, 0);
}

function renderLoop(time) {
    ctx.setTransform(1,0,0,1,0,0);
    ctx.clearRect(0, 0, w, h);
    var x = Math.cos(time / 500) * (cw - 80) + cw;
    var y = Math.sin(time / 600) * (ch - 80) + ch;
    var rad = Math.sin(time / 333) * 10 + 30;
    drawImageCircle(imgPat, minRadius, x, y, rad);
    requestAnimationFrame(renderLoop);
}
canvas {
   background: #888;
   border: 2px solid black;
}
<canvas id="canvas" width="256" height="256"></canvas>


0
投票

如果您的用例支持,我建议使用具有预先裁剪的圆形透明度的 png 以提高性能并避免编写此代码。

但是让我们继续回答您的问题来自评论。我觉得这与带有两个二次曲线的 Canvas 剪辑图像有足够的不同,可以添加新答案,但该线程显示了一般方法:在带有 context.clip() 的路径之后使用

context.arc
,然后以 
context.drawImage
 完成.

由于您还要绘制其他内容,请用

context.save()

context.restore()
 包裹您的剪辑,以防止您的剪辑影响您之后绘制的所有内容。

这是一个最小的例子:

const canvas = document.createElement("canvas"); canvas.height = canvas.width = 100; const {width: w, height: h} = canvas; document.body.appendChild(canvas); const ctx = canvas.getContext("2d"); const img = new Image(); img.src = `https://picsum.photos/${w}/${h}`; img.decode().then(() => { ctx.fillRect(10, 0, 20, 20); // normal drawing before save() ctx.save(); ctx.beginPath(); ctx.arc(w / 2, h / 2, w / 2, 0, Math.PI * 2); ctx.clip(); ctx.drawImage(img, 0, 0); ctx.restore(); ctx.fillRect(0, 10, 20, 20); // back to normal drawing after restore() });

如果您有多个图像,我建议使用

在一个函数中加载图像中描述的承诺。

© www.soinside.com 2019 - 2024. All rights reserved.