Не могу нарисовать холст, используя палец в мобильных браузерах

Вот код:

canvas = document.getElementById("canvas");
ctx = canvas.getContext('2d');
tmp_ctx = element[0].getContext('2d');

element.bind('mousemove touchmove', function(event){
                if(drawing){
                    if(event.offsetX!==undefined){
                        currentX = event.offsetX;
                        currentY = event.offsetY;
                    } else {
                        currentX = event.layerX - event.currentTarget.offsetLeft;
                        currentY = event.layerY - event.currentTarget.offsetTop;
                    }

                    if(currentTool=="karandaw" || currentTool=="lastik" || currentTool=="brush"){
                        drawPen(lastX, lastY, currentX, currentY);
                    }
                    if(currentTool=="oval"){
                        console.log("drawing ellipse");
                        drawEllipse(initX, initY, event.offsetX, event.offsetY);
                    }
                    if(currentTool=="crop"){
                        drawCrop(initX, initY, event.offsetX, event.offsetY);
                    }

                    lastX = currentX;
                    lastY = currentY;
                }
            });


function drawEllipse(x1, y1, x2, y2){
                console.log(tmp_ctx);
                tmp_ctx.clearRect(0, 0, tmp_canvas.width, tmp_canvas.height);
                tmp_ctx.setLineDash([1,0]);
                tmp_ctx.beginPath();
                tmp_ctx.lineWidth = 3;
                tmp_ctx.strokeStyle = 'rgba('+curColor.r+','+curColor.g+','
                    +curColor.b+','+curColor.a+')';
                ctx.globalCompositeOperation='source-over';
                var mpx = (x1+x2)/2;
                var mpy = (y1+y2)/2;
                tmp_ctx.ellipse(mpx, mpy, Math.abs(mpx-x1), Math.abs(mpy-y1), 0, 0, Math.PI*2);
                tmp_ctx.stroke();
                tmp_ctx.closePath();
            }

Он хорошо работает в браузерах ПК. Но на мобильных телефонах ничего не рисуется, хотя журналы показывают, что эти функции для рисования называются.

Есть ли какая-то ошибка?

HTML:

 <div id="canvas_container">
                        <canvas id="canvas"></canvas>
                        <canvas id="tmp_canvas" drawing></canvas>
                        <div id="canvas_area_1" ng-show="canvas_area_1"></div>
                        <div id="canvas_area_2" ng-show="canvas_area_2"></div>
                        <div id="canvas_area_3" ng-show="canvas_area_3"></div>
                    </div>

РЕДАКТИРОВАТЬ 1: Chrome версии на мобильном телефоне было 47

РЕДАКТИРОВАТЬ 2: я использую директиву (с угловой), чтобы сделать это. Вот полный код директивы:

app.directive("drawing", function(){
    return {
        restrict: "A",
        link: function(scope, element){
            canvas = document.getElementById("canvas");
            ctx = canvas.getContext('2d');
            tmp_ctx = element[0].getContext('2d');

            var drawing = false;
            var lastX;
            var lastY;
            var initX;
            var initY;

            //--- MOUSE EVENTS

            $(element).bind('mousedown touchstart', function(event){
                if(event.offsetX!==undefined){
                    initX = event.offsetX;
                    lastX = initX;
                    initY = event.offsetY;
                    lastY = initY;
                } else {
                    initX = event.layerX - event.currentTarget.offsetLeft;
                    initY = event.layerY - event.currentTarget.offsetTop;
                    lastX = initX;
                    lastY = initY;
                }

                if(currentTool=="karandaw" || currentTool=="lastik" || currentTool=="brush"){
                    drawPen(initX, initY, initX, initY);
                    appendDraw();
                }

                img_old = ctx.getImageData(0, 0, canvas.width, canvas.height);
                imageData = img_old;
                drawing = true;
                tmp_ctx.clearRect(0, 0, tmp_canvas.width, tmp_canvas.height);
            });

            $(element).bind('mousemove touchmove', function(event){
                if(drawing){
                    if(event.offsetX!==undefined){
                        currentX = event.offsetX;
                        currentY = event.offsetY;
                    } else {
                        currentX = event.layerX - event.currentTarget.offsetLeft;
                        currentY = event.layerY - event.currentTarget.offsetTop;
                    }

                    if(currentTool=="karandaw" || currentTool=="lastik" || currentTool=="brush"){
                        drawPen(lastX, lastY, currentX, currentY);
                    }
                    if(currentTool=="oval"){
                        console.log("drawing ellipse");
                        drawEllipse(initX, initY, event.offsetX, event.offsetY);
                    }
                    if(currentTool=="crop"){
                        drawCrop(initX, initY, event.offsetX, event.offsetY);
                    }

                    lastX = currentX;
                    lastY = currentY;
                }
            });

            $(element).bind('mouseup touchleave touchcancel', function(event){
                drawing = false;
                appendDraw();
            });

            //--- CHECK AND APPEND DRAW

            function appendDraw(){
                startedPainting = true;

                if(currentTool=="oval"){
                    drawEllipse(initX, initY, event.offsetX, event.offsetY);

                    if(drawPart=="oval_1"){
                        if(isCorrectDraw("oval_1", initX, initY, event.offsetX, event.offsetY)){
                            var mainScope = angular.element(document.getElementById("main")).scope();
                            startedPainting = false;
                            mainScope.$apply(function (){mainScope.level_3_part_17();});
                        }else{
                            tmp_ctx.clearRect(0, 0, tmp_canvas.width, tmp_canvas.height);
                            return;
                        }
                    }

                    if(drawPart=="oval_2"){
                        if(isCorrectDraw("oval_2", initX, initY, event.offsetX, event.offsetY)){
                            var mainScope = angular.element(document.getElementById("main")).scope();
                            mainScope.$apply(function (){mainScope.level_3_part_14();});
                            startedPainting = false;
                        }else{
                            startedPainting = false;
                            tmp_ctx.clearRect(0, 0, tmp_canvas.width, tmp_canvas.height);
                            var mainScope = angular.element(document.getElementById("main")).scope();
                            mainScope.$apply(function (){mainScope.level_3_part_12();});
                            return;
                        }
                    }

                    if(drawPart=="oval_3"){
                        if(isCorrectDraw("oval_3", initX, initY, event.offsetX, event.offsetY)){
                            var mainScope = angular.element(document.getElementById("main")).scope();
                            mainScope.$apply(function (){mainScope.level_3_part_8();});
                            startedPainting = false;
                        }else{
                            tmp_ctx.clearRect(0, 0, tmp_canvas.width, tmp_canvas.height);
                            return;
                        }
                    }
                }else
                if( (currentTool=="karandaw" || currentTool=="brush") && drawPart=="pen"){
                    if(isCorrectDraw("pen")){
                        penCount++;
                        if(penCount==4){
                            var mainScope = angular.element(document.getElementById("main")).scope();
                            mainScope.$apply(function (){mainScope.level_3_part_23();});
                        }
                    }else{
                        tmp_ctx.clearRect(0, 0, tmp_canvas.width, tmp_canvas.height);
                        return;
                    }

                }else
                if(currentTool=="crop"){
                    cropRight = false;
                    var mainScope = angular.element(document.getElementById("main")).scope();
                    mainScope.$apply(function (){mainScope.enableCut();});
                    return;
                }else
                if(currentTool=="cut"){
                    enableUndo();
                    return;
                }else
                if(currentTool=="paint"){
                    paintAt(event.offsetX, event.offsetY);
                    enableUndo();
                    return;
                }

                ctx.drawImage(tmp_canvas, 0, 0);
                fillTransparent();
                tmp_ctx.clearRect(0, 0, tmp_canvas.width, tmp_canvas.height);
                enableUndo();
            }

            function enableUndo(){
                img_new=ctx.getImageData(0, 0, canvas.width, canvas.height);
                $("#level_3_otmenit_nazhata").css("display", "none");
            }

            //--- CHECK IF DRAWN CORRECTLY

            function isCorrectDraw(part, x1, y1, x2, y2){
                var result = true;
                if(part=="oval_3"){
                    oval_3_rx = (x1+x2)/2;
                    oval_3_ry = (y1+y2)/2;
                    oval_3_w = Math.abs(oval_3_rx-x1)*2;
                    oval_3_h = Math.abs(oval_3_ry-y1)*2;
                    if(oval_3_h>oval_3_maxH() || oval_3_w>oval_3_maxW() ||
                        oval_3_h<oval_3_minH() || oval_3_w<oval_3_minW())
                        result = false;
                }else
                if(part=="oval_2"){
                    oval_2_rx = (x1+x2)/2;
                    oval_2_ry = (y1+y2)/2;
                    oval_2_w = Math.abs(oval_2_rx-x1)*2;
                    oval_2_h = Math.abs(oval_2_ry-y1)*2;
                    if(oval_2_h>oval_2_maxH() || oval_2_w>oval_2_maxW() ||
                        oval_2_h<oval_2_minH() || oval_2_w<oval_2_minW() || !isAllowedShift(oval_2_rx, oval_3_rx))
                        result = false;
                }else
                if(part=="oval_1"){
                    oval_1_rx = (x1+x2)/2;
                    oval_1_ry = (y1+y2)/2;
                    oval_1_w = Math.abs(oval_1_rx-x1)*2;
                    oval_1_h = Math.abs(oval_1_ry-y1)*2;
                    if(oval_1_h>oval_1_maxH() || oval_1_w>oval_1_maxW() ||
                        oval_1_h<oval_1_minH() || oval_1_w<oval_1_minW() || !isAllowedShift(oval_1_rx, oval_3_rx))
                        result = false;
                }else
                if(part=="pen"){
                    var xshift = (oval_1_h>oval_1_w)? oval_1_h : oval_1_w;

                    if(Math.abs(oval_1_rx-lastX)>xshift || lastY>oval_1_ry )
                        result = false;
                }
                return result;
            }

            function isPointInEllipse(x, y, rx, ry, h, k){
                var a = (x-h)*(x-h);
                var b = (y-k)*(y-k);
                var c = rx*rx;
                var d = ry*ry;
                if( (a/c +  b/d) <=1)
                    return true;
                return false;
            }

            //--- DRAWING FUNCTIONS

            function fillTransparent(){
                var imgd = ctx.getImageData(0, 0, canvas.width, canvas.height);
                var pix = imgd.data;

                for (var i = 0, n = pix.length; i < n; i += 4) {
                    if(pix[i]==0 && pix[i+1]==0 && pix[i+2]==0 && pix[i+3]==0){
                        pix[i] = 255;
                        pix[i+1] = 255;
                        pix[i+2] = 255;
                        pix[i+3] = 255;
                    }
                }
                ctx.putImageData(imgd, 0, 0);
            }

            function drawPen(x1, y1, x2, y2){
                tmp_ctx.lineCap = "round";
                tmp_ctx.lineJoin = "round";
                tmp_ctx.setLineDash([1,0]);
                tmp_ctx.beginPath();
                tmp_ctx.lineWidth = currentTool=="karandaw" ? 5 : 10;
                if(currentTool=="lastik"){
                    ctx.globalCompositeOperation='destination-out';
                    tmp_ctx.strokeStyle = "white";
                }else{
                    ctx.globalCompositeOperation='source-over';
                    tmp_ctx.strokeStyle = 'rgba('+curColor.r+','+curColor.g+','
                        +curColor.b+','+curColor.a+')';
                }

                if(x1==x2 && y1==y2)
                    x1++;

                tmp_ctx.moveTo(x1,y1);
                tmp_ctx.lineTo(x2,y2);
                tmp_ctx.stroke();
                tmp_ctx.closePath();
            }

            function drawEllipse(x1, y1, x2, y2){
                console.log(tmp_ctx);
                tmp_ctx.clearRect(0, 0, tmp_canvas.width, tmp_canvas.height);
                tmp_ctx.setLineDash([1,0]);
                tmp_ctx.beginPath();
                tmp_ctx.lineWidth = 3;
                tmp_ctx.strokeStyle = 'rgba('+curColor.r+','+curColor.g+','
                    +curColor.b+','+curColor.a+')';
                ctx.globalCompositeOperation='source-over';
                var mpx = (x1+x2)/2;
                var mpy = (y1+y2)/2;
                tmp_ctx.ellipse(mpx, mpy, Math.abs(mpx-x1), Math.abs(mpy-y1), 0, 0, Math.PI*2);
                tmp_ctx.stroke();
                tmp_ctx.closePath();
            }

            function drawCrop(x1, y1, x2, y2){
                cutX = Math.min(x1, x2); cutY = Math.min(y1, y2); cutW = Math.abs(x2-x1); cutH = Math.abs(y2-y1);
                tmp_ctx.clearRect(0, 0, tmp_canvas.width, tmp_canvas.height);
                tmp_ctx.setLineDash([10]);
                tmp_ctx.beginPath();
                tmp_ctx.lineWidth = 1;
                tmp_ctx.strokeStyle = "rgba(0,0,0,1)";
                ctx.globalCompositeOperation='source-over';
                tmp_ctx.rect(cutX, cutY, cutW, cutH);
                tmp_ctx.stroke();
                tmp_ctx.closePath();
            }

            //---PAINT BUCKET

            function paintAt(startX, startY){
                imageData = ctx.getImageData(0, 0, canvas.width, canvas.height);

                var pixelPos = (startY * canvas.width + startX) * 4,
                    r = imageData.data[pixelPos],
                    g = imageData.data[pixelPos + 1],
                    b = imageData.data[pixelPos + 2],
                    a = imageData.data[pixelPos + 3];

                if (r === curColor.r && g === curColor.g && b === curColor.b && a === curColor.a){
                    return;
                }

                floodFill(startX, startY, r, g, b, a);

                clear_ctx(0, 0, canvas.width, canvas.height);
                ctx.putImageData(imageData, 0, 0);
            }

            function matchStartColor(pixelPos, startR, startG, startB, startA){
                var r = imageData.data[pixelPos],
                    g = imageData.data[pixelPos + 1],
                    b = imageData.data[pixelPos + 2],
                    a = imageData.data[pixelPos + 3];

                if (r === startR && g === startG && b === startB && a === startA)
                    return true;

                if (r === curColor.r && g === curColor.g && b === curColor.b && a === curColor.a)
                    return false;

                return false;
            }

            function colorPixel(pixelPos){
                imageData.data[pixelPos] = curColor.r;
                imageData.data[pixelPos + 1] = curColor.g;
                imageData.data[pixelPos + 2] = curColor.b;
                imageData.data[pixelPos + 3] = 255;
            }

            function floodFill(startX, startY, startR, startG, startB, startA){
                var newPos, x, y, pixelPos, reachLeft, reachRight,
                    drawingBoundTop = 0,
                    pixelStack = [[startX, startY]];

                while (pixelStack.length){
                    newPos = pixelStack.pop();
                    x = newPos[0];
                    y = newPos[1];

                    pixelPos = (y * canvas.width + x) * 4;

                    while(y-- >= drawingBoundTop && matchStartColor(pixelPos, startR, startG, startB, startA))
                        pixelPos -= canvas.width * 4;

                    pixelPos += canvas.width * 4;
                    y += 1;
                    reachLeft = false;
                    reachRight = false;

                    while (y++ < canvas.height-1 && matchStartColor(pixelPos, startR, startG, startB, startA)){
                        colorPixel(pixelPos);

                        if (x > 0){
                            if (matchStartColor(pixelPos - 4, startR, startG, startB, startA)){
                                if (!reachLeft){
                                    pixelStack.push([x - 1, y]);
                                    reachLeft = true;
                                }
                            } else if (reachLeft){
                                reachLeft = false;
                            }
                        }
                        if (x < canvas.width-1){
                            if (matchStartColor(pixelPos + 4, startR, startG, startB, startA)){
                                if (!reachRight){
                                    pixelStack.push([x + 1, y]);
                                    reachRight = true;
                                }
                            } else if (reachRight){
                                reachRight = false;
                            }
                        }

                        pixelPos += canvas.width * 4;
                    }
                }
            }//floodFill
        }
    };
});

1 ответ

Я думаю, что bind(), который вы имели в виду, это jquery.bind(), а не нативный javascript .bind();

если вы идете чистый путь JavaScript, попробуйте использовать addEventListener.

function moveHandler(event){
    if(drawing){
       if(event.offsetX!==undefined){
           currentX = event.offsetX;
           //.... next code
       }         
    }
}
element.addEventListener('touchmove', moveHandler);
element.addEventListener('mousemove', moveHandler);

instead of $(element).bind
Другие вопросы по тегам