旋转图像和像素碰撞检测

Rotating Images & Pixel Collision Detection

我在这个 plunker.

里有这个游戏

当剑不旋转时,一切正常(您可以通过取消注释行 221 并注释掉 222-223 来检查)。当它们像上面的 plunker 一样旋转时,碰撞效果不佳。

我想那是因为 "getImageData" 记住了旧图像,但我认为一遍又一遍地重新计算是一件很昂贵的事情。

是否有更好的方法来旋转我的图像并使其正常工作?还是我必须重新计算他​​们的像素图?

罪魁祸首代码:

for (var i = 0; i < monsters.length; i++) {
    var monster = monsters[i];
    if (monster.ready) {
        if (imageCompletelyOutsideCanvas(monster, monster.monsterImage)) {
            monster.remove = true;
        }
        //else {
        //ctx.drawImage(monster.monsterImage, monster.x, monster.y);
        drawRotatedImage(monster.monsterImage, monster.x, monster.y, monster);
        monster.rotateCounter += 0.05;
        //}
    }
}

几何解

通过更快的几何解决方案来做到这一点。

最简单的解决方案是线段与圆相交算法。

线段.

一条线的起点和终点可以用多种方式描述。在这种情况下,我们将使用开始和结束坐标。

var line = {
    x1 : ?,
    y1 : ?,
    x2 : ?,
    y2 : ?,
}

圆形

圆是由它的位置和半径描述的

var circle = {
   x : ?,
   y : ?,
   r : ?,
}

圆线段相交

下面介绍我是如何测试圆线段碰撞的。我不知道是否有更好的方法(很可能有),但这对我很有帮助并且可靠,但需要注意的是线段必须有长度,圆必须有面积。如果您不能保证这一点,那么您必须在代码中添加检查以确保您不会被零除。

因此,为了测试一条线是否截取圆,我们首先找出线上最近的点有多远(注意线是无限长的,而线段有长度、起点和终点)

// a quick convertion of vars to make it easier to read.
var x1 = line.x1;
var y1 = line.y1;
var x2 = line.x2;
var y2 = line.y2;

var cx = circle.x;
var cy = circle.y;
var r = circle.r;

测试结果,如果发生碰撞,则为真。

var result; // the result of the test

将直线转换为向量。

var vx = x2 - x1;  // convert line to vector
var vy = y2 - y1;
var d2 = (vx * vx + vy * vy);  // get the length squared

获取直线上近点到圆的单位距离。单位距离是从 0 到 1(含)的数字,表示沿点向量的距离。如果该值小于 0 则该点在向量之前,如果大于 1 则该点超过末尾。

这个我是凭记忆知道的,忘记概念了。它是线向量与从线段起点到圆心的向量除以线向量长度平方的点积。

// dot product of two vectors is v1.x * v2.x + v1.y * v2.y over v1 length squared 
u =  ((cx - x1) * vx + (cy - y1) * vy) / d2;

现在使用单位位置,通过将线向量乘以单位距离加上线段起始位置,得到直线上最接近圆的点的实际坐标。

 // get the closest point
var  xx = x1 + vx * u;
var  yy = y1 + vy * u;

现在我们在直线上有一个点,我们使用毕达哥拉斯平方根计算与圆的距离。

// get the distance from the circle center
var d =  Math.hypot(xx - cx, yy - cy);    

现在,如果直线(不是线段)与圆相交,则距离将等于或小于圆半径。否则没有拦截。

if(d > r){ //is the distance greater than the radius
    result = false;  // no intercept
} else { // else we need some more calculations

要确定线段是否截取了圆,我们需要找到该线穿过圆周上的两个点。我们有半径和圆与直线的距离。由于与直线的距离始终成直角,因此我们有一个直角三角形,其中 hypot 是半径,一侧是找到的距离。

算出三角形缺失的长度。 UPDATE 在 "update" 下的答案底部查看代码的改进版本,它使用单位长度而不是归一化线向量。

// ld for line distance is the square root of the hyp subtract side squared
var ld = Math.sqrt(r * r - d * d);

现在将该距离添加到我们在直线 xx 上找到的点,yy 通过将线向量除以来对线向量进行归一化(使线向量成为一个单位长)它的长度,然后乘以上面找到的距离

var len = Math.sqrt(d2); // get the line vector length
var nx = (vx / len) * ld;      
var ny = (vy / len) * ld;      

有些人可能会看到我可以使用单位长度并跳过一些计算。是的,但我可能会因为重写演示而烦恼,所以将保持原样

现在通过将新向量加减到离圆最近的直线上的点来得到截点

ix1 = xx + nx; // the point furthest alone the line 
iy1 = xx + ny;
ix2 = xx - nx; // the point in the other direction
iy2 = xx - ny;

现在我们有了这两个点,我们可以计算出它们是否在线段中,但计算它们在原始线向量上的单位距离,使用点积除以平方距离。

    var u1 =  ((ix1 - x1) * vx + (iy1 - y1) * vy) / d2;
    var u2 =  ((ix2 - x1) * vx + (iy1 - y1) * vy) / d2; 

现在做一些简单的测试,看看这些点的单位postion是否在线段上

    if(u1 < 0){  // is the forward intercept befor the line segment start
        result = false;  // no intercept            
    }else
    if(u2 > 1){ // is the rear intercept after the line end
        result = false;  // no intercept            
    } else {
        // though the line segment may not have intercepted the circle
        // circumference if we have got to here it must meet the conditions
        // of touching some part of the circle.
        result = true;
    }
}

演示

一如既往,这里有一个展示逻辑的演示。圆圈以鼠标为中心。如果圆圈接触到一些测试线,它们会变红。它还将显示圆周确实穿过线的点。如果位于线段内,则点为红色;如果位于线段外,则点为绿色。这些点可用于添加效果或其他什么

我今天很懒,所以这是直接从我的图书馆里拿出来的。 注意,有机会我会post改进数学。

更新

我改进了算法,用单位长度计算圆的周长相交,减少了很多代码。我也把它添加到演示中了。

从直线距离小于圆半径的点开始

            // get the unit distance to the intercepts
            var ld = Math.sqrt(r * r - d * d) / Math.sqrt(d2);

            // get that points unit distance along the line
            var u1 =  u + ld; 
            var u2 =  u - ld; 
            if(u1 < 0){  // is the forward intercept befor the line
                result = false;  // no intercept
            }else
            if(u2 > 1){  // is the backward intercept past the end of the line
                result = false;  // no intercept
            }else{
                result = true;
            }
        }

var demo = function(){
    
    // the function described in the answer with extra stuff for the demo
    // at the bottom you will find the function being used to test circle intercepts.
    
    
    /** GeomDependancies.js begin **/
        
    // for speeding up calculations.
    // usage may vary from descriptions. See function for any special usage notes
    var data = {
        x:0,   // coordinate
        y:0,
        x1:0,   // 2nd coordinate if needed
        y1:0,
        u:0,   // unit length
        i:0,   // index
        d:0,   // distance
        d2:0,  // distance squared
        l:0,   // length
        nx:0,  // normal vector
        ny:0,
        result:false, // boolean result
    }
    // make sure hypot is suported
    if(typeof Math.hypot !== "function"){
        Math.hypot = function(x, y){ return Math.sqrt(x * x + y * y);};
    }
    /** GeomDependancies.js end **/
    
    /** LineSegCircleIntercept.js begin **/
    // use data properties
    // result  // intercept bool for intercept
    // x, y    // forward intercept point on line ** 
    // x1, y1  // backward intercept point on line
    // u       // unit distance of intercept mid point
    // d2      // line seg length squared
    // d       // distance of closest point on line from circle
    // i       // bit 0 on for forward intercept on segment 
    //         // bit 1 on for backward intercept
    // ** x = null id intercept points dont exist
    var lineSegCircleIntercept = function(ret, x1, y1, x2, y2, cx, cy, r){
    var vx, vy, u, u1, u2, d, ld, len, xx, yy;
        vx = x2 - x1;  // convert line to vector
        vy = y2 - y1;
        ret.d2 = (vx * vx + vy * vy);
        
        // get the unit distance of the near point on the line
        ret.u = u =  ((cx - x1) * vx + (cy - y1) * vy) / ret.d2;
        xx = x1 + vx * u; // get the closest point
        yy = y1 + vy * u;
        
        // get the distance from the circle center
        ret.d = d =  Math.hypot(xx - cx, yy - cy);    
        if(d <= r){ // line is inside circle
            // get the distance to the two intercept points
            ld = Math.sqrt(r * r - d * d) / Math.sqrt(ret.d2);

            // get that points unit distance along the line
            u1 =  u + ld; 
            if(u1 < 0){  // is the forward intercept befor the line
                ret.result = false;  // no intercept
                return ret;
            }
            u2 =  u - ld; 
            if(u2 > 1){  // is the backward intercept past the end of the line
                ret.result = false;  // no intercept
                return ret;
            }
            ret.i = 0;
            if(u1 <= 1){
                ret.i += 1;
                // get the forward point line intercepts the circle
                ret.x = x1 + vx * u1;  
                ret.y = y1 + vy * u1;
            }else{
                ret.x = x2;
                ret.y = y2;
                
            }
            if(u2 >= 0){
                ret.x1 = x1 + vx * u2;  
                ret.y1 = y1 + vy * u2;
                ret.i += 2;
            }else{
                ret.x1 = x1;
                ret.y1 = y1;
            }
            
            // tough the points of intercept may not be on the line seg
            // the closest point to the must be on the line segment
            ret.result = true;
            return ret;
            
        }
        ret.x = null; // flag that no intercept found at all;
        ret.result = false;  // no intercept
        return ret;
            
    }
    /** LineSegCircleIntercept.js end **/
    

    // mouse and canvas functions for this demo.

    /** fullScreenCanvas.js begin **/
    var canvas = (function(){
        var canvas = document.getElementById("canv");
        if(canvas !== null){
            document.body.removeChild(canvas);
        }
        // creates a blank image with 2d context
        canvas = document.createElement("canvas"); 
        canvas.id = "canv";    
        canvas.width = window.innerWidth;
        canvas.height = window.innerHeight; 
        canvas.style.position = "absolute";
        canvas.style.top = "0px";
        canvas.style.left = "0px";
        canvas.style.zIndex = 1000;
        canvas.ctx = canvas.getContext("2d"); 
        document.body.appendChild(canvas);
        return canvas;
    })();
    var ctx = canvas.ctx;
    
    /** fullScreenCanvas.js end **/
    /** MouseFull.js begin **/
    
    var canvasMouseCallBack = undefined;  // if needed
    var mouse = (function(){
        var mouse = {
            x : 0, y : 0, w : 0, alt : false, shift : false, ctrl : false,
            interfaceId : 0, buttonLastRaw : 0,  buttonRaw : 0,
            over : false,  // mouse is over the element
            bm : [1, 2, 4, 6, 5, 3], // masks for setting and clearing button raw bits;
            getInterfaceId : function () { return this.interfaceId++; }, // For UI functions
            startMouse:undefined,
        };
        function mouseMove(e) {
            var t = e.type, m = mouse;
            m.x = e.offsetX; m.y = e.offsetY;
            if (m.x === undefined) { m.x = e.clientX; m.y = e.clientY; }
            m.alt = e.altKey;m.shift = e.shiftKey;m.ctrl = e.ctrlKey;
            if (t === "mousedown") { m.buttonRaw |= m.bm[e.which-1];
            } else if (t === "mouseup") { m.buttonRaw &= m.bm[e.which + 2];
            } else if (t === "mouseout") { m.buttonRaw = 0; m.over = false;
            } else if (t === "mouseover") { m.over = true;
            } else if (t === "mousewheel") { m.w = e.wheelDelta;
            } else if (t === "DOMMouseScroll") { m.w = -e.detail;}
            if (canvasMouseCallBack) { canvasMouseCallBack(m.x, m.y); }
            e.preventDefault();
        }
        function startMouse(element){
            if(element === undefined){
                element = document;
            }
            "mousemove,mousedown,mouseup,mouseout,mouseover,mousewheel,DOMMouseScroll".split(",").forEach(
            function(n){element.addEventListener(n, mouseMove);});
            element.addEventListener("contextmenu", function (e) {e.preventDefault();}, false);
        }
        mouse.mouseStart = startMouse;
        return mouse;
    })();
    if(typeof canvas === "undefined"){
        mouse.mouseStart(canvas);
    }else{
        mouse.mouseStart();
    }
    /** MouseFull.js end **/
    
    // helper function
    function drawCircle(ctx,x,y,r,col,col1,lWidth){
        if(col1){
            ctx.lineWidth = lWidth;
            ctx.strokeStyle = col1;
        }
        if(col){
            ctx.fillStyle = col;
        }
        
        ctx.beginPath();
        ctx.arc( x, y, r, 0, Math.PI*2);
        if(col){
            ctx.fill();
        }
        if(col1){
            ctx.stroke();
        }
    }
    
    // helper function
    function drawLine(ctx,x1,y1,x2,y2,col,lWidth){
        ctx.lineWidth = lWidth;
        ctx.strokeStyle = col;
        ctx.beginPath();
        ctx.moveTo(x1,y1);
        ctx.lineTo(x2,y2);
        ctx.stroke();
    }
    var h = canvas.height;
    var w = canvas.width;
    var unit = Math.ceil(Math.sqrt(Math.hypot(w, h)) / 32);
    const U80 = unit * 80;
    const U60 = unit * 60;
    const U40 = unit * 40;
    const U10 = unit * 10;
    var lines = [
        {x1 : U80, y1 : U80, x2 : w /2, y2 : h - U80},
        {x1 : w - U80, y1 : U80, x2 : w /2, y2 : h - U80},
        {x1 : w / 2 - U10, y1 : h / 2 - U40, x2 : w /2, y2 : h/2 + U10 * 2},
        {x1 : w / 2 + U10, y1 : h / 2 - U40, x2 : w /2, y2 : h/2 + U10 * 2},
    ];
    
    function update(){
        var i, l;
        ctx.clearRect(0, 0, w, h);
        
        drawCircle(ctx, mouse.x, mouse.y, U60, undefined, "black", unit * 3);
        drawCircle(ctx, mouse.x, mouse.y, U60, undefined, "yellow", unit * 2);
        for(i = 0; i < lines.length; i ++){
            l = lines[i]
            drawLine(ctx, l.x1, l.y1, l.x2, l.y2, "black" , unit * 3)
            drawLine(ctx, l.x1, l.y1, l.x2, l.y2, "yellow" , unit * 2)
            
            // test the lineSegment circle
            data = lineSegCircleIntercept(data,  l.x1, l.y1, l.x2, l.y2, mouse.x, mouse.y, U60);
            // if there is a result display the result
            if(data.result){
                drawLine(ctx, l.x1, l.y1, l.x2, l.y2, "red" , unit * 2)
                if((data.i & 1) === 1){
                    drawCircle(ctx, data.x, data.y, unit * 4, "white", "red", unit );
                }else{
                    drawCircle(ctx, data.x, data.y, unit * 2, "white", "green", unit );
                }
                if((data.i & 2) === 2){
                    drawCircle(ctx, data.x1, data.y1, unit * 4, "white", "red", unit );
                }else{
                    drawCircle(ctx, data.x1, data.y1, unit * 2, "white", "green", unit );
                }
            }
        }
        requestAnimationFrame(update);
    }
    
    update();
}
// resize if needed by just starting again
window.addEventListener("resize",demo);

// start the demo
demo();

...这里是如何在移动和旋转剑时找到剑刃线

首先找到原始剑刃的顶点并将它们保存在一个数组中。

var pts=[{x:28,y:42},{x:69,y:3},{x:83,y:1},{x:83,y:19},{x:42,y:57}];

剑旋转时,每个剑刃顶点都会围绕旋转点旋转。在你的例子中,旋转点是图像的中心。

  • 灰色矩形是图像的矩形边框
  • 蓝点是一个剑尖(在刀尖)
  • 绿点在图像中心(==旋转点)
  • 绿线是图像中心到顶点的距离
  • 蓝色圆圈是刀尖旋转 360 度时所遵循的路径
  • 绿线会根据图像的旋转改变角度。

你可以像这样计算任意旋转角度的叶尖位置:

// [cx,cy] = the image centerpoint (== the rotation point)
// [vx,vy] = the coordinate position of the blade tip
// Calculate the distance and the angle between the 2 points 
var dx=vx-cx;
var dy=vy-cy;
var distance=Math.sqrt(dx*dx+dy*dy);
var originalAngle=Math.atan2(dy,dx);

// rotationAngle = the angle the image has been rotated expressed in radians
var rotatedX = cx + distance * Math.cos(originalAngle + rotationAngle);
var rotatedY = cy + distance * Math.sin(originalAngle + rotationAngle);

这是在移动和旋转时跟踪叶片顶点的示例代码和演示:

var canvas=document.getElementById("canvas");
var ctx=canvas.getContext("2d");
var cw=canvas.width;
var ch=canvas.height;
function reOffset(){
  var BB=canvas.getBoundingClientRect();
  offsetX=BB.left;
  offsetY=BB.top;        
}
var offsetX,offsetY;
reOffset();
window.onscroll=function(e){ reOffset(); }
window.onresize=function(e){ reOffset(); }

var isDown=false;
var startX,startY;

var sword={
    img:null,
    rx:0,
    ry:0,
    angle:0,
    pts:[{x:28,y:42},{x:69,y:3},{x:83,y:1},{x:83,y:19},{x:42,y:57}],
    // precalculated properties -- for efficiency
    radii:[],
    angles:[],
    halfWidth:0,
    halfHeight:0,
    //
    initImg:function(img){
        var PI2=Math.PI*2;
        this.img=img;
        this.halfWidth=img.width/2;
        this.halfHeight=img.height/2;
        for(var i=0;i<this.pts.length;i++){
            var dx=this.halfWidth-this.pts[i].x;
            var dy=this.halfHeight-this.pts[i].y;
            this.radii[i]=Math.sqrt(dx*dx+dy*dy);
            this.angles[i]=((Math.atan2(dy,dx)+PI2)%PI2)-Math.PI;
        }
    },
    // draw sword with translation & rotation
    draw:function(){
        var img=this.img;
        var rx=this.rx;
        var ry=this.ry;
        var angle=this.angle;
        ctx.translate(rx,ry);
        ctx.rotate(angle);
        ctx.drawImage(img,-this.halfWidth,-this.halfHeight);
        ctx.rotate(-angle);
        ctx.translate(-rx,-ry);
    },
    // recalc this.pts after translation & rotation
    calcTrxPts:function(){
        var trxPts=[];
        for(var i=0;i<this.pts.length;i++){
            var r=this.radii[i];
            var ptangle=this.angles[i]+this.angle;
            trxPts[i]={
                x:this.rx+r*Math.cos(ptangle),
                y:this.ry+r*Math.sin(ptangle)
            };
        }
        return(trxPts);
    },
}

// load image & initialize sword object & draw scene
var img=new Image();
img.onload=function(){
    // set initial sword properties
    sword.initImg(img);
    sword.rx=150;
    sword.ry=75;
    sword.angle=0; //(Math.PI/8);

    // draw scene
    drawAll();

    // listen for mouse events
    $("#canvas").mousedown(function(e){handleMouseDown(e);});
    $("#canvas").mousemove(function(e){handleMouseMove(e);});
    $("#canvas").mouseup(function(e){handleMouseUpOut(e);});
    $("#canvas").mouseout(function(e){handleMouseUpOut(e);});

    // listen for mousewheel events
    $("#canvas").on('DOMMouseScroll mousewheel',function(e){
        e.preventDefault();
        e.stopPropagation();
        var e=e || window.event; // old IE support
        sign=((e.originalEvent.wheelDelta||e.originalEvent.detail*-1)>0)?1:-1;
        sword.angle+=Math.PI/45*sign;
        drawAll();
    });
}
img.src = "";


/////////////////////
// helper functions
/////////////////////

function drawAll(){
    ctx.clearRect(0,0,cw,ch);
    sword.draw();
    drawHitArea();
}

function drawHitArea(){
    // lines
    var trxPts=sword.calcTrxPts();
    ctx.beginPath();
    ctx.moveTo(trxPts[0].x,trxPts[0].y);
    for(var i=1;i<trxPts.length;i++){
        ctx.lineTo(trxPts[i].x,trxPts[i].y);
    }
    ctx.closePath();
    ctx.strokeStyle='red';
    ctx.stroke();
    // dots
    for(var i=0;i<trxPts.length;i++){
        ctx.beginPath();
        ctx.arc(trxPts[i].x,trxPts[i].y,3,0,Math.PI*2);
        ctx.closePath();
        ctx.fillStyle='blue';
        ctx.fill();
    }
}

function getClosestPointOnLineSegment(line,x,y) {
    //
    lerp=function(a,b,x){ return(a+x*(b-a)); };
    var dx=line.x1-line.x0;
    var dy=line.y1-line.y0;
    var t=((x-line.x0)*dx+(y-line.y0)*dy)/(dx*dx+dy*dy);
    var lineX=lerp(line.x0, line.x1, t);
    var lineY=lerp(line.y0, line.y1, t);
    return({x:lineX,y:lineY,isOnSegment:(t>=0 && t<=1)});
};


function handleMouseDown(e){
  // tell the browser we're handling this event
  e.preventDefault();
  e.stopPropagation();
  
  startX=parseInt(e.clientX-offsetX);
  startY=parseInt(e.clientY-offsetY);

  // Put your mousedown stuff here
  isDown=true;
}

function handleMouseUpOut(e){
  // tell the browser we're handling this event
  e.preventDefault();
  e.stopPropagation();
  // clear the isDragging flag
  isDown=false;
}

function handleMouseMove(e){
  if(!isDown){return;}
  // tell the browser we're handling this event
  e.preventDefault();
  e.stopPropagation();

  // calc distance moved since last drag
  mouseX=parseInt(e.clientX-offsetX);
  mouseY=parseInt(e.clientY-offsetY);
  var dx=mouseX-startX;
  var dy=mouseY-startY;
  startX=mouseX;
  startY=mouseY;

  // drag the sword to new position
  sword.rx+=dx;
  sword.ry+=dy;
  drawAll();
}
body{ background-color: ivory; }
#canvas{border:1px solid red; }
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.9.1/jquery.min.js"></script>
<h6>Drag sword and<br>Rotate sword using mousewheel inside canvas<br>Red "collision" lines follow swords translation & rotation.</h6>
<h5></h5>
<canvas id="canvas" width=300 height=300></canvas>