Поворот изображения и обнаружение столкновения пикселей

У меня есть эта игра в этом поршне.

Когда мечи не вращаются, все работает нормально (вы можете проверить это, раскомментировав строки 221 и закомментировав 222-223). Когда они вращаются, как в поршне выше, столкновение не работает хорошо.

Я думаю, это потому, что "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;
        //}
    }
}

2 ответа

Решение

Геометрическое решение

Чтобы сделать это через более быстрое геометрическое решение.

Самое простое решение - это отрезок с алгоритмом пересечения окружности.

Отрезок.

Строка имеет начало и конец, описанные различными способами. В этом случае мы будем использовать начальную и конечную координаты.

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

Чтобы определить, пересекает ли отрезок отрезок окружность, нам нужно найти две точки на окружности окружности, которые линия пересекла. У нас есть радиус и расстояние, на котором круг находится от линии. Поскольку расстояние от линии всегда находится под прямым углом, у нас есть прямоугольный треугольник с гипотезой, являющейся радиусом, а одна сторона - найденным расстоянием.

Определите недостающую длину треугольника. ОБНОВЛЕНИЕ смотрите улучшенную версию кода отсюда внизу ответа в разделе "обновление", он использует длины единиц, а не нормализует вектор строки.

// 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; 

Теперь несколько простых тестов, чтобы увидеть, находятся ли единичные позиции этих точек на отрезке

    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;
    }
}

демонстрация

Как всегда, здесь представлена ​​демонстрация, демонстрирующая логику в действии. Круг центрирован на мышке. Есть несколько тестовых линий, которые станут красными, если круг коснется их. Это также покажет точку, где окружность круга пересекает линию. Точка будет красной, если она находится в отрезке, или зеленой, если она находится снаружи. Эти точки можно использовать, чтобы добавить эффекты или что нет

Я ленивый сегодня, так что это прямо из моей библиотеки. Обратите внимание, я опубликую улучшенную математику, когда у меня будет шанс.

Обновить

Я улучшил алгоритм, используя длину единицы для вычисления пересечений окружности окружности, устраняя много кода. Я добавил это и к демке.

От точки, где расстояние от линии меньше радиуса круга

            // 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>

Другие вопросы по тегам