撤消/重做不正常,缩放后绘画也不能正常工作

时间:2016-04-19 08:50:17

标签: javascript jquery canvas html5-canvas undo-redo

我正在尝试使用撤消和重做功能实现一个paint bucket工具。问题是undo和redo第一次正常工作,但是当我多次撤消重做时,代码失败了。任何人都可以帮我解决这个问题吗?此外,缩放功能正常,但缩放后绘画无法正常工作。这是我的完整代码。你可以复制粘贴,它将在你的最后工作。

<!DOCTYPE html>
<html>
    <head>
        <title>Painitng</title>
        <style>
            body {
                width: 100%;
                height: auto;
                text-align: center;
            }
            .colorpick {
                widh: 100%;
                height: atuo;
            }
            .pick {
                display: inline-block;
                width: 30px;
                height: 30px;
                margin: 5px;
                cursor: pointer;
            }
            canvas {
                border: 2px solid silver;
            }
        </style>
    </head>
    <body>
        <button id="zoomin">Zoom In</button>
        <button id="zoomout">Zoom Out</button>
        <button onclick="undo()">Undo</button>
        <button onclick="redo()">Redo</button>
        <div id="canvasDiv"></div>
        <script type="text/javascript" src="http://ajax.googleapis.com/ajax/libs/jquery/1.4.2/jquery.js"></script> 
        <script type="text/javascript">
            var colorYellow = {
                r: 255,
                g: 207,
                b: 51
            };
            var context;
            var canvasWidth = 500;
            var canvasHeight = 500;
            var myColor = colorYellow;
            var curColor = myColor;
            var outlineImage = new Image();
            var backgroundImage = new Image();
            var drawingAreaX = 0;
            var drawingAreaY = 0;
            var drawingAreaWidth = 500;
            var drawingAreaHeight = 500;
            var colorLayerData;
            var outlineLayerData;
            var totalLoadResources = 2;
            var curLoadResNum = 0;
            var undoarr = new Array();
            var redoarr = new Array();
            var uc = 0;
            var rc = 0;

            // Clears the canvas.
            function clearCanvas() {
                context.clearRect(0, 0, context.canvas.width, context.canvas.height);
            }

            function undo() {
                if (undoarr.length <= 0)
                    return;

                if (uc==0) {
                    redoarr.push(undoarr.pop());
                    uc = 1;
                }
                var a = undoarr.pop();
                colorLayerData = a;
                redoarr.push(a);
                clearCanvas();
                context.putImageData(a, 0, 0);
                context.drawImage(backgroundImage, 0, 0, canvasWidth, canvasHeight);
                context.drawImage(outlineImage, 0, 0, drawingAreaWidth, drawingAreaHeight);
                console.log(undoarr);
            }

            function redo() {
                if (redoarr.length <= 0)
                    return;
                if (rc==0) {
                    undoarr.push(redoarr.pop());
                    rc = 1;
                }
                var a = redoarr.pop();
                colorLayerData = a;
                undoarr.push(a);
                clearCanvas();
                context.putImageData(a, 0, 0);
                context.drawImage(backgroundImage, 0, 0, canvasWidth, canvasHeight);
                context.drawImage(outlineImage, 0, 0, drawingAreaWidth, drawingAreaHeight);
                console.log(redoarr);
            }
            // Draw the elements on the canvas
            function redraw() {
                uc = 0;
                rc = 0;
                var locX,
                        locY;

                // Make sure required resources are loaded before redrawing
                if (curLoadResNum < totalLoadResources) {
                    return; // To check if images are loaded successfully or not.
                }

                clearCanvas();
                // Draw the current state of the color layer to the canvas
                context.putImageData(colorLayerData, 0, 0);

                undoarr.push(context.getImageData(0, 0, canvasWidth, canvasHeight));
                console.log(undoarr);
                redoarr = new Array();
                // Draw the background
                context.drawImage(backgroundImage, 0, 0, canvasWidth, canvasHeight);

                // Draw the outline image on top of everything. We could move this to a separate 
                //   canvas so we did not have to redraw this everyime.
                context.drawImage(outlineImage, 0, 0, drawingAreaWidth, drawingAreaHeight);


            }
            ;

            function matchOutlineColor(r, g, b, a) {

                return (r + g + b < 100 && a === 255);
            }
            ;

            function matchStartColor(pixelPos, startR, startG, startB) {

                var r = outlineLayerData.data[pixelPos],
                        g = outlineLayerData.data[pixelPos + 1],
                        b = outlineLayerData.data[pixelPos + 2],
                        a = outlineLayerData.data[pixelPos + 3];

                // If current pixel of the outline image is black
                if (matchOutlineColor(r, g, b, a)) {
                    return false;
                }

                r = colorLayerData.data[pixelPos];
                g = colorLayerData.data[pixelPos + 1];
                b = colorLayerData.data[pixelPos + 2];

                // If the current pixel matches the clicked color
                if (r === startR && g === startG && b === startB) {
                    return true;
                }

                // If current pixel matches the new color
                if (r === curColor.r && g === curColor.g && b === curColor.b) {
                    return false;
                }

                return true;
            }
            ;

            function colorPixel(pixelPos, r, g, b, a) {
                colorLayerData.data[pixelPos] = r;
                colorLayerData.data[pixelPos + 1] = g;
                colorLayerData.data[pixelPos + 2] = b;
                colorLayerData.data[pixelPos + 3] = a !== undefined ? a : 255;
            }
            ;

            function floodFill(startX, startY, startR, startG, startB) {
                var newPos,
                        x,
                        y,
                        pixelPos,
                        reachLeft,
                        reachRight,
                        drawingBoundLeft = drawingAreaX,
                        drawingBoundTop = drawingAreaY,
                        drawingBoundRight = drawingAreaX + drawingAreaWidth - 1,
                        drawingBoundBottom = drawingAreaY + drawingAreaHeight - 1,
                        pixelStack = [[startX, startY]];

                while (pixelStack.length) {

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

                    // Get current pixel position
                    pixelPos = (y * canvasWidth + x) * 4;

                    // Go up as long as the color matches and are inside the canvas
                    while (y >= drawingBoundTop && matchStartColor(pixelPos, startR, startG, startB)) {
                        y -= 1;
                        pixelPos -= canvasWidth * 4;
                    }

                    pixelPos += canvasWidth * 4;
                    y += 1;
                    reachLeft = false;
                    reachRight = false;

                    // Go down as long as the color matches and in inside the canvas
                    while (y <= drawingBoundBottom && matchStartColor(pixelPos, startR, startG, startB)) {
                        y += 1;

                        colorPixel(pixelPos, curColor.r, curColor.g, curColor.b);

                        if (x > drawingBoundLeft) {
                            if (matchStartColor(pixelPos - 4, startR, startG, startB)) {
                                if (!reachLeft) {
                                    // Add pixel to stack
                                    pixelStack.push([x - 1, y]);
                                    reachLeft = true;
                                }

                            } else if (reachLeft) {
                                reachLeft = false;
                            }
                        }

                        if (x < drawingBoundRight) {
                            if (matchStartColor(pixelPos + 4, startR, startG, startB)) {
                                if (!reachRight) {
                                    // Add pixel to stack
                                    pixelStack.push([x + 1, y]);
                                    reachRight = true;
                                }
                            } else if (reachRight) {
                                reachRight = false;
                            }
                        }

                        pixelPos += canvasWidth * 4;
                    }
                }
            }
            ;

            // Start painting with paint bucket tool starting from pixel specified by startX and startY
            function paintAt(startX, startY) {

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

                if (r === curColor.r && g === curColor.g && b === curColor.b) {
                    // Return because trying to fill with the same color
                    return;
                }

                if (matchOutlineColor(r, g, b, a)) {
                    // Return because clicked outline
                    return;
                }

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

                redraw();
            }
            ;

            // Add mouse event listeners to the canvas
            function createMouseEvents() {

                $('#canvas').mousedown(function (e) {
                    // Mouse down location
                    var mouseX = e.pageX - this.offsetLeft,
                            mouseY = e.pageY - this.offsetTop;

                    if ((mouseY > drawingAreaY && mouseY < drawingAreaY + drawingAreaHeight) && (mouseX <= drawingAreaX + drawingAreaWidth)) {
                        paintAt(mouseX, mouseY);
                    }
                });
            }
            ;

            resourceLoaded = function () {

                curLoadResNum += 1;
                //if (curLoadResNum === totalLoadResources) {
                createMouseEvents();
                redraw();
                //}
            };

            function start() {

                var canvas = document.createElement('canvas');
                canvas.setAttribute('width', canvasWidth);
                canvas.setAttribute('height', canvasHeight);
                canvas.setAttribute('id', 'canvas');
                document.getElementById('canvasDiv').appendChild(canvas);

                if (typeof G_vmlCanvasManager !== "undefined") {
                    canvas = G_vmlCanvasManager.initElement(canvas);
                }
                context = canvas.getContext("2d");
                backgroundImage.onload = resourceLoaded();
                backgroundImage.src = "images/t1.png";

                outlineImage.onload = function () {
                    context.drawImage(outlineImage, drawingAreaX, drawingAreaY, drawingAreaWidth, drawingAreaHeight);

                    try {
                        outlineLayerData = context.getImageData(0, 0, canvasWidth, canvasHeight);
                    } catch (ex) {
                        window.alert("Application cannot be run locally. Please run on a server.");
                        return;
                    }
                    clearCanvas();
                    colorLayerData = context.getImageData(0, 0, canvasWidth, canvasHeight);
                    resourceLoaded();
                };
                outlineImage.src = "images/d.png";
            }
            ;

            getColor = function () {

            };

        </script> 
        <script type="text/javascript"> $(document).ready(function () {
                start();
            });</script> 
        <script language="javascript">
            $('#zoomin').click(function () {
                if ($("#canvas").width()==500){
                $("#canvas").width(750);
                $("#canvas").height(750);
                var ctx = canvas.getContext("2d");
                ctx.drawImage(backgroundImage, 0, 0, 749, 749);
                ctx.drawImage(outlineImage, 0, 0, 749, 749);
                redraw();
                 } else if ($("#canvas").width()==750){

                $("#canvas").width(1000);
                $("#canvas").height(1000);
                var ctx = canvas.getContext("2d");
                ctx.drawImage(backgroundImage, 0, 0, 999, 999);
                ctx.drawImage(outlineImage, 0, 0, 999, 999);
                redraw();
                 }
            });
            $('#zoomout').click(function () {
                if ($("#canvas").width() == 1000) {

                $("#canvas").width(750);
                $("#canvas").height(750);
                var ctx = canvas.getContext("2d");
                ctx.drawImage(backgroundImage, 0, 0, 749, 749);
                ctx.drawImage(outlineImage, 0, 0, 749, 749);
                redraw();
                } else if ($("#canvas").width() == 750) {

                $("#canvas").width(500);
                $("#canvas").height(500);
                var ctx = canvas.getContext("2d");
                ctx.drawImage(backgroundImage, 0, 0, 499, 499);
                ctx.drawImage(outlineImage, 0, 0, 499, 499);
                redraw();
                }
            });
        </script>
        <div class="colorpick">
            <div class="pick" style="background-color:rgb(150, 0, 0);" onclick="hello(this.style.backgroundColor);"></div>
            <div class="pick" style="background-color:rgb(0, 0, 152);" onclick="hello(this.style.backgroundColor);"></div>
            <div class="pick" style="background-color:rgb(0, 151, 0);" onclick="hello(this.style.backgroundColor);"></div>
            <div class="pick" style="background-color:rgb(255, 0, 5);" onclick="hello(this.style.backgroundColor);"></div>
            <div class="pick" style="background-color:rgb(255, 255, 0);" onclick="hello(this.style.backgroundColor);"></div>
            <div class="pick" style="background-color:rgb(0, 255, 255);" onclick="hello(this.style.backgroundColor);"></div>
            <div class="pick" style="background-color:rgb(255, 0, 255);" onclick="hello(this.style.backgroundColor);"></div>
            <div class="pick" style="background-color:rgb(255, 150, 0);" onclick="hello(this.style.backgroundColor);"></div>
            <div class="pick" style="background-color:rgb(255, 0, 150);" onclick="hello(this.style.backgroundColor);"></div>
            <div class="pick" style="background-color:rgb(0, 255, 150);" onclick="hello(this.style.backgroundColor);"></div>
            <div class="pick" style="background-color:rgb(150, 0, 255);" onclick="hello(this.style.backgroundColor);"></div>
            <div class="pick" style="background-color:rgb(0, 150, 255);" onclick="hello(this.style.backgroundColor);"></div>
        </div>
        <script>
            function hello(e) {
                var rgb = e.replace(/^(rgb|rgba)\(/, '').replace(/\)$/, '').replace(/\s/g, '').split(',');
                myColor.r = parseInt(rgb[0]);
                myColor.g = parseInt(rgb[1]);
                myColor.b = parseInt(rgb[2]);
                curColor = myColor;
                console.log(curColor);
            }
        </script>
    </body>
</html>

2 个答案:

答案 0 :(得分:1)

此函数matchOutlineColor包含4个代表RGBA颜色的数字。

  

红色,绿色,蓝色,Alpha(颜色有多透明)

RGBA颜色范围为0-255,因此从0(无颜色)到255(全彩色),白色为 rgba(255,255,255,255),黑色为 rgba(0,0) ,0,255)透明为 rgba(0,0,0,0)

此代码不检查颜色是否为黑色,只是加在一起的红色+绿色+黄色至少小于100(总共750个)。我怀疑该功能检查颜色是否为深色。

例如,这一切都将成立:

&#13;
&#13;
<div style="background-color:rgba(99,0,0,255)">Dark RED</div>
<div style="background-color:rgba(0,99,0,255)">Dark GREEN</div>
<div style="background-color:rgba(0,0,99,255)">Dark BLUE</div>
&#13;
&#13;
&#13;

如果要检查边框是否为黑色,可以将功能更改为

function matchOutlineColorBlack(r, g, b, a) {
    //Ensures red + green + blue is none
    return (r + g + b == 0 && a === 255);
};

function matchOutlineColorWhite(r, g, b, a) {
    //Checks that color is white (255+255+255=750)
    return (r + g + b == 750 && a === 255);
};

答案 1 :(得分:1)

帆布尺寸&amp;州历史

画布尺寸

如果您曾经浏览过DOM,您会注意到许多元素都将高度和宽度作为属性,高度和宽度都作为样式属性。

对于画布,这些有两种不同的含义。所以我们创建一个画布。

var canvas = document.createElement("canvas");

现在可以设置画布元素的宽度和高度。这定义了画布图像中的像素数(分辨率)

canvas.width = 500;
canvas.height = 500;

默认情况下,当图像(画布只是图像)显示在DOM中时,它以一对一的像素大小显示。这意味着对于图像中的每个像素,页面上都有一个像素。

您可以通过设置画布样式宽度和高度

来更改此设置
canvas.style.width = "1000px"; // Note you must add the unit type "px" in this case
canvas.style.width = "1000px";

这不会改变画布分辨率,只会改变显示尺寸。现在,对于画布中的每个像素,它在页面上占用4个像素。

使用鼠标绘制到画布时会出现问题,因为鼠标坐标的屏幕像素不再与画布分辨率相匹配。

解决这个问题。并作为OP代码的一个例子。您需要重新缩放鼠标坐标以匹配画布分辨率。这已添加到OP mousedown事件侦听器中。它首先获得显示宽度/高度,然后是分辨率宽度和高度。它通过除以显示宽度/高度来标准化鼠标坐标。这使得鼠标坐标在0 <=鼠标<0的范围内。 1然后我们乘以得到画布像素坐标。由于像素需要位于整数位置(整数),因此必须对结果进行处理。

// assuming that the mouseX and mouseY are the mouse coords.
if(this.style.width){   // make sure there is a width in the style 
                        // (assumes if width is there then height will be too
    var w = Number(this.style.width.replace("px",""));  // warning this will not work if size is not in pixels
    var h = Number(this.style.height.replace("px","")); // convert the height to a number
    var pixelW = this.width;  // get  the canvas resolution
    var pixelH = this.height;
    mouseX = Math.floor((mouseX / w) * pixelW); // convert the mouse coords to pixel coords
    mouseY = Math.floor((mouseY / h) * pixelH);
}

这将解决您的缩放问题。但是看看你的代码,它是一团糟,你不应该每次都在搜索nodetree,重新获取上下文。我很惊讶它有效,但那可能是Jquery(我不知道,因为我从来没有使用它)或者可能是你在其他地方渲染。

州历史

计算机程序的当前状态是定义当前状态的所有条件和数据。当您保存某个状态时保存状态,并在加载时恢复状态。

历史只是一种保存和加载状态的方法,而不会在文件系统中乱七八糟。它有一些约定表明统计信息存储为堆栈。第一个是最后一个,它有一个重做堆栈,允许你重做以前的撤消,但保持正确的状态,因为状态依赖于以前的状态,重做只能从关联状态重做。因此,如果您撤消然后绘制某些内容,则会使任何现有重做状态无效,并且应该将它们转储。

此外,已保存的状态(无论是磁盘还是撤消堆栈)必须与当前状态分离。如果您对当前状态进行了更改,则不希望这些更改影响已保存的状态。

我认为这是你出错的地方OP,因为你在撤消时使用colorLayerData来填充(绘画)或重做你使用撤销/重做缓冲区中保留的引用数据的地方当你绘制时,你实际上仍在更改撤消缓冲区中的数据。

历史记录经理

这是一个通用状态管理器,可以满足任何撤消/重做需求,您所要做的就是确保将当前状态收集到一个对象中。

为了帮助我写一个简单的历史经理。它有两个缓冲区作为堆栈,一个用于undos,另一个用于redos。它还保持当前状态,这是它所知道的最新状态。

当您推送到历史记录管理器时,它将采用它所知道的当前状态并将其推送到撤消堆栈,保存当前状态,并使任何重做数据无效(使重做数组长度为0)

撤消时会将当前状态推送到重做堆栈,从撤消堆栈弹出状态并将其置于当前状态,然后返回当前状态。

重做时,它会将当前状态推送到撤销堆栈,从重做堆栈弹出状态并将其置于当前状态,然后它将返回当前状态。

从状态管理器返回状态的副本非常重要,这样您就不会无意中更改缓冲区中存储的数据。

你可能会问。 &#34;为什么州政府经理不能确保数据是副本?&#34;一个很好的问题,但这不是一个州经理的角色,它保存了州,它必须这样做,无论它需要保存什么,它本质上完全不知道它存储的数据的含义。这样它就可以用于图像,文本,游戏状态,任何东西,就像文件系统一样,它不能(不应该)意识到其含义,因此知道如何创建有意义的副本。您推送到状态管理器的数据只是像素数据的单个参考(64位长),或者您可以推送像素数据的每个字节,它不知道差异。

另外OP我已经向状态管理器添加了一些UI控件。这允许它显示其当前状态即禁用并启用撤消重做按钮。对于提供反馈的良好UI设计而言,它始终是非常重要的。

代码

您需要对代码进行以下所有更改才能使用历史记录管理器。您可以这样做,或者只是将其作为指南并编写自己的指南。我在检测到您的错误之前写了这个。如果这是唯一的错误,那么您可能只需要更改。

   // your old code (from memory)
   colorLayerData = undoArr.pop(); 
   context.putImageData(colorLayerData, 0, 0);  


   // the fix same applies to redo and just makes a copy rather than use 
   // the reference that is still stored in the undoe buff
   context.putImageData(undoArr, 0, 0);   // put the undo onto the canvas
   colorLayerData = context.getImageData(0, 0, canvasWidth, canvaHeight); 

删除撤消/重做的所有代码。

将页面顶部的撤消/重做按钮更改为,使用单个功能处理这两个事件。

    <button id = "undo-button" onclick="history('undo')">Undo</button>
    <button id = "redo-button" onclick="history('redo')">Redo</button>

将以下两个函数添加到代码

        function history(command){ // handles undo/redo button events.
            var data;
            if(command === "redo"){
                data = historyManager.redo(); // get data for redo
            }else
            if(command === "undo"){
                data = historyManager.undo(); // get data for undo
            }
            if(data !== undefined){ // if data has been found
                setColorLayer(data); // set the data
            }
        }

        // sets colour layer and creates copy into colorLayerData
        function setColorLayer(data){
            context.putImageData(data, 0, 0);  
            colorLayerData = context.getImageData(0, 0, canvasWidth, canvasHeight);
            context.drawImage(backgroundImage, 0, 0, canvasWidth, canvasHeight);
            context.drawImage(outlineImage, 0, 0, drawingAreaWidth, drawingAreaHeight);            
        }

在重绘功能中,您已经替换了用于撤消的内容,并在同一位置添加此行。这会将当前状态保存在历史记录管理器中。

           historyManager.push(context.getImageData(0, 0, canvasWidth, canvasHeight));

在启动功能中,您必须将UI元素添加到状态管理器。这取决于您,可以忽略,如果未定义,则统计管理器将忽略它们。

            if(historyManager !== undefined){
                // only for visual feedback and not required for the history manager to function.
                historyManager.UI.assignUndoButton(document.querySelector("#undo-button"));
                historyManager.UI.assignRedoButton(document.querySelector("#redo-button"));
            }

当然,历史管理员自己。它封装了数据,因此除了通过接口提供外,您无法访问其内部状态。

historyManager(hM)API

  • hM.UI ui经理只是更新并分配按钮 已禁用/已启用状态
  • hM.UI.assignUndoButton(element)设置撤消元素
  • hM.UI.assignRedoButton(element)设置重做元素
  • nM.UI.update()更新按钮状态以反映当前状态 内部国家。所有内部状态都会自动调用 仅在您更改自我重做/撤消按钮统计数据时才需要
  • hM.reset()重置历史记录管理器,清除所有堆栈和当前保存的状态。加载或创建新项目时调用此方法。
  • nM.push(data)将提供的数据添加到历史记录中。
  • nM.undo()获取以前的历史状态并返回存储的数据。如果没有数据,那么这将返回undefined。
  • nM.redo()获取下一个历史记录状态并返回存储的数据。如果没有数据,那么这将返回undefined。

自调用函数创建历史管理器,通过变量historyManager

访问接口
var historyManager = (function (){  // Anon for private (closure) scope
    var uBuffer = []; // this is undo buff
    var rBuffer = []; // this is redo buff
    var currentState = undefined; // this holds the current history state
    var undoElement = undefined;
    var redoElement = undefined;
    var manager = {
        UI : {  // UI interface just for disable and enabling redo undo buttons
            assignUndoButton : function(element){
                undoElement = element;
                this.update();
            },
            assignRedoButton : function(element){
                redoElement = element;
                this.update();
            },
            update : function(){
                if(redoElement !== undefined){
                    redoElement.disabled = (rBuffer.length === 0);
                }
                if(undoElement !== undefined){
                    undoElement.disabled = (uBuffer.length === 0);                                
                }
            }
        },
        reset : function(){
            uBuffer.length = 0;
            rBuffer.length = 0;
            currentState = undefined;
            this.UI.update();
        },
        push : function(data){
            if(currentState !== undefined){
                uBuffer.push(currentState);                        
            }
            currentState = data;
            rBuffer.length = 0;
            this.UI.update();
        },
        undo : function(){
           if(uBuffer.length > 0){
               if(currentState !== undefined){
                    rBuffer.push(currentState);                        
                }
                currentState = uBuffer.pop();
            }
            this.UI.update();
            return currentState; // return data or unfefined
        },
        redo : function(){
            if(rBuffer.length > 0){
               if(currentState !== undefined){
                    uBuffer.push(currentState);                        
                }
                currentState = rBuffer.pop();
            }
            this.UI.update();    
            return currentState;
        },
    }
    return manager;
})();

这将解决您的缩放问题和撤消问题。祝你的项目好运。