(function() { var nodeEnv = typeof require !== 'undefined' && typeof process !== 'undefined'; var __module = nodeEnv ? module : {exports:{}}; var __filename = 'preview-scripts/assets/Script/Game.js'; var __require = nodeEnv ? function (request) { return cc.require(request); } : function (request) { return __quick_compile_project__.require(request, __filename); }; function __define (exports, require, module) { if (!nodeEnv) {__quick_compile_project__.registerModule(__filename, module);}"use strict"; cc._RF.push(module, '1c26dVSok5E7o2PwD2sGBJa', 'Game'); // Script/Game.ts "use strict"; // Learn TypeScript: // - https://docs.cocos.com/creator/manual/en/scripting/typescript.html // Learn Attribute: // - https://docs.cocos.com/creator/manual/en/scripting/reference/attributes.html // Learn life-cycle callbacks: // - https://docs.cocos.com/creator/manual/en/scripting/life-cycle-callbacks.html var __extends = (this && this.__extends) || (function () { var extendStatics = function (d, b) { extendStatics = Object.setPrototypeOf || ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) || function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; }; return extendStatics(d, b); }; return function (d, b) { extendStatics(d, b); function __() { this.constructor = d; } d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __()); }; })(); var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) { var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d; if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc); else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r; return c > 3 && r && Object.defineProperty(target, key, r), r; }; Object.defineProperty(exports, "__esModule", { value: true }); var DrawingBoard_1 = require("./DrawingBoard"); var _a = cc._decorator, ccclass = _a.ccclass, property = _a.property; var GameState; (function (GameState) { GameState[GameState["drawing"] = 1] = "drawing"; GameState[GameState["erasing"] = 2] = "erasing"; })(GameState || (GameState = {})); var Game = /** @class */ (function (_super) { __extends(Game, _super); function Game() { var _this = _super !== null && _super.apply(this, arguments) || this; _this.drawNode = null; _this.captureCamera = null; _this.mainCamera = null; _this.db = null; _this.gameState = GameState.drawing; _this.texture = null; _this.prePos = cc.Vec2.ZERO; _this.startPos = cc.Vec2.ZERO; _this.lastColor = cc.Color.BLUE; _this.errColor = cc.Color.RED; _this.lastLineWidth = 1; _this.history = []; _this.touchId = -1; _this.touchScale = false; return _this; } Game.prototype.start = function () { var _this = this; this.initDb(); this.initTexture(); this.initRead(); setTimeout(function () { _this.drawNode.on("touchstart", _this.onTouchStart, _this); _this.drawNode.on("touchmove", _this.onTouchMove, _this); _this.drawNode.on("touchend", _this.onTouchEnd, _this); _this.drawNode.on("touchcancel", _this.onTouchEnd, _this); }, 2000); }; Game.prototype.initDb = function () { //创建一个画板(需传入画板尺寸,将自动初始化) this.db = new DrawingBoard_1.default(this.drawNode.width, this.drawNode.height); //设置画板的绘图颜色(每次绘制前都可以重新设置) this.lastLineWidth = 15; this.db.setLineWidth(this.lastLineWidth); // this.db.setColor(this.lastColor.r, this.lastColor.g, this.lastColor.b, this.lastColor.a); //线条端点以圆角结尾 this.db.setLineCircleEnd(true); }; Game.prototype.initTexture = function () { this.texture = new cc.RenderTexture(); this.texture.initWithSize(this.drawNode.width, this.drawNode.height, cc.RenderTexture.DepthStencilFormat.RB_FMT_S8); var spf = new cc.SpriteFrame(this.texture); this.drawNode.getComponent(cc.Sprite).spriteFrame = spf; }; Game.prototype.initRead = function () { this.targetCamera = this.node.getChildByName("tagCamera").getComponent(cc.Camera); var rander = new cc.RenderTexture(); rander.initWithSize(this.node.width, this.node.height, cc.RenderTexture.DepthStencilFormat.RB_FMT_S8); this.targetCamera.targetTexture = rander; this.targetCamera.render(); console.log("完成"); }; Game.prototype.onTouchStart = function (e) { //将触摸位置作为线条的起点 //画板中使用的坐标系,与图片坐标系一样,原点在左上角,X轴向右为正,Y轴向下为正 //所以Y轴坐标应反过来, 这里用getLocationInView而不是getLocation this.touchId = e.getID(); if (this.touchId == 1) { this.touchScale = true; return; } var pos = e.getLocation(); this.prePos = this.convertToDrawNodePos(pos); this.startPos = this.convertToDrawNodePos(pos); this.db.moveTo(this.prePos.x, this.prePos.y); }; Game.prototype.onTouchMove = function (e) { var touches = e.getTouches(); var touch1 = touches[0]; var delta1 = touch1.getDelta(); var pos = e.getLocation(); var pos1 = this.convertToDrawNodePos(touch1.getLocation()); var dst = this.startPos.sub(pos1).mag(); // this.label.string = touches.length + ""; if (touches.length == 1 && this.touchId < 1 && !this.touchScale && dst > 7) { // alert("不该进来"); this.prePos = this.convertToDrawNodePos(pos); var jg = this.pd(e); this.changeColor(jg); if (this.gameState == GameState.drawing) { //从上一次绘制线条后的终点开始向鼠标当前位置绘制线条 this.db.lineTo(this.prePos.x, this.prePos.y); } else if (this.gameState == GameState.erasing) { // 橡皮擦 this.db.circle(this.prePos.x, this.prePos.y, 10); } //每次画板中的数据有变化后,及时将数据应用到贴图上,在屏幕上显示出来 this.drawToImg(); } else if (touches.length == 2) { var touch1 = touches[0], touch2 = touches[1]; var delta1 = touch1.getDelta(), delta2 = touch2.getDelta(); var touchPoint1 = this.node.parent.convertToNodeSpaceAR(touch1.getLocation()); var touchPoint2 = this.node.parent.convertToNodeSpaceAR(touch2.getLocation()); var distance = touchPoint1.sub(touchPoint2); var delta = delta1.sub(delta2); var scale = 1; if (Math.abs(distance.x) > Math.abs(distance.y)) { scale = (distance.x + delta.x) / distance.x * this.node.scale; } else { scale = (distance.y + delta.y) / distance.y * this.node.scale; } if (scale > 2) scale = 2; this.node.scale = scale <= 0.1 ? 0.1 : scale; } }; Game.prototype.onTouchEnd = function (e) { this.touchId = e.getID(); if (this.touchId == 1) this.touchScale = false; this.addHistory(); }; Game.prototype.pd = function (event) { var cha = 2; var pos = event.getLocation(); var jg = false; for (var i = -cha; i < cha; i++) { var postion = cc.v2(); postion.x = pos.x + i; for (var j = -cha; j < cha; j++) { postion.y = pos.y + j; // console.log("检测点:",postion.x,postion.y); var img = this.getGraphisData(postion); if ((img[0] != 255 && img[1] != 255 && img[2] != 255)) { jg = true; j = 10000; i = 10000; } } } // return jg; }; Game.prototype.convertToDrawNodePos = function (worldPos) { var pos = this.drawNode.convertToNodeSpaceAR(worldPos); pos.x += this.drawNode.width * this.drawNode.anchorX; pos.y += this.drawNode.height * this.drawNode.anchorY; pos.y = this.drawNode.height - pos.y; return pos; }; Game.prototype.addHistory = function () { var copy = this.db.copyData(); var ucopy = new Uint8Array(copy); this.history.push({ data: ucopy }); // cc.log('历史步骤: ', this.history.length); }; Game.prototype.drawToImg = function () { //获取画板中绘制的图案数据 var data = this.db.getData(); //将数据传递给贴图对象 this.texture.initWithData(data, cc.Texture2D.PixelFormat.RGBA8888, this.db.width, this.db.height); }; Game.prototype.changeColor = function (red) { if (!red) this.db.setColor(this.errColor.r, this.errColor.g, this.errColor.b, this.errColor.a); else this.db.setColor(this.lastColor.r, this.lastColor.g, this.lastColor.b, this.lastColor.a); }; Game.prototype.getGraphisData = function (point) { var Uint8 = new Uint8Array(4); Uint8 = this.targetCamera.targetTexture.readPixels(Uint8, point.x, point.y, 1, 1); return Uint8; }; Game.prototype.onBtnDraw = function () { this.db.setLineWidth(this.lastLineWidth); this.db.setColor(this.lastColor.r, this.lastColor.g, this.lastColor.b, this.lastColor.a); this.gameState = GameState.drawing; }; Game.prototype.onBtnErase = function () { this.db.setLineWidth(this.lastLineWidth * 3); // 橡皮擦的颜色不能是(0,0,0,0),因为这样会和DrawingBoard里的默认颜色相同导致绘制跳过 this.db.setColor(255, 255, 255, 0); this.gameState = GameState.erasing; }; Game.prototype.onBtnClear = function () { this.db.reset(); this.drawToImg(); this.history.splice(0, this.history.length); }; Game.prototype.onBtnRevoke = function () { this.history.pop(); if (this.history.length) { var data = this.history[this.history.length - 1].data; this.db.setData(data.buffer); this.texture.initWithData(this.db.getData(), cc.Texture2D.PixelFormat.RGBA8888, this.db.width, this.db.height); } else { this.onBtnClear(); } cc.log('历史记录剩余: ', this.history.length); }; Game.prototype.onBtnSave = function () { var _this = this; if (cc.sys.isBrowser) { var width = this.drawNode.width; var height = this.drawNode.height; this.captureCamera.enabled = true; var texture = new cc.RenderTexture(); texture.initWithSize(width, height, cc.RenderTexture.DepthStencilFormat.RB_FMT_S8); this.captureCamera.targetTexture = texture; var canvas = document.createElement('canvas'); canvas.width = width; canvas.height = height; var ctx = canvas.getContext('2d'); this.captureCamera.render(); var data = texture.readPixels(); // write the render data var rowBytes = width * 4; for (var row = 0; row < height; row++) { var srow = height - 1 - row; var imageData = ctx.createImageData(width, 1); var start = srow * width * 4; for (var i = 0; i < rowBytes; i++) { imageData.data[i] = data[start + i]; } ctx.putImageData(imageData, 0, row); } // var dataUrl = canvas.toDataURL('image/png'); // cc.log('iamge-base64:', dataUrl); var saveLink = document.createElementNS('http://www.w3.org/1999/xhtml', 'a'); saveLink.href = dataUrl; saveLink.download = String(Date.now()) + '.png'; var event = document.createEvent('MouseEvents'); event.initMouseEvent('click', true, false, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null); saveLink.dispatchEvent(event); this.scheduleOnce(function (t) { _this.captureCamera.enabled = false; }, 0.1); } else { cc.warn('暂时只支持web端保存图片'); } }; Game.prototype.update = function (dt) { }; __decorate([ property(cc.Node) ], Game.prototype, "drawNode", void 0); __decorate([ property(cc.Camera) ], Game.prototype, "captureCamera", void 0); __decorate([ property(cc.Camera) ], Game.prototype, "mainCamera", void 0); Game = __decorate([ ccclass ], Game); return Game; }(cc.Component)); exports.default = Game; cc._RF.pop(); } if (nodeEnv) { __define(__module.exports, __require, __module); } else { __quick_compile_project__.registerModuleFunc(__filename, function () { __define(__module.exports, __require, __module); }); } })(); //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["assets\\Script\\Game.ts"],"names":[],"mappings":"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","file":"","sourceRoot":"/","sourcesContent":["// Learn TypeScript:\n//  - https://docs.cocos.com/creator/manual/en/scripting/typescript.html\n// Learn Attribute:\n//  - https://docs.cocos.com/creator/manual/en/scripting/reference/attributes.html\n// Learn life-cycle callbacks:\n//  - https://docs.cocos.com/creator/manual/en/scripting/life-cycle-callbacks.html\n\nimport DrawingBoard from \"./DrawingBoard\";\n\nconst { ccclass, property } = cc._decorator;\n\nenum GameState {\n    drawing = 1,\n    erasing = 2,\n}\n\n@ccclass\nexport default class Game extends cc.Component {\n    @property(cc.Node)\n    drawNode: cc.Node = null;\n\n    @property(cc.Camera)\n    captureCamera: cc.Camera = null;\n\n    @property(cc.Camera)\n    mainCamera: cc.Camera = null;\n\n    private targetCamera: cc.Camera;\n    private db: DrawingBoard = null;\n    private gameState: GameState = GameState.drawing;\n    private texture: cc.RenderTexture = null;\n    private prePos: cc.Vec2 = cc.Vec2.ZERO;\n    private startPos: cc.Vec2 = cc.Vec2.ZERO;\n    private lastColor: cc.Color = cc.Color.BLUE;\n    private errColor: cc.Color = cc.Color.RED;\n    private lastLineWidth: number = 1;\n    private history: any[] = [];\n    private touchId = -1;\n    private touchScale = false;\n  \n\n    start() {\n        this.initDb();\n        this.initTexture();\n        this.initRead();\n\n        setTimeout(() => {\n            this.drawNode.on(\"touchstart\", this.onTouchStart, this);\n            this.drawNode.on(\"touchmove\", this.onTouchMove, this);\n            this.drawNode.on(\"touchend\", this.onTouchEnd, this);\n            this.drawNode.on(\"touchcancel\", this.onTouchEnd, this);\n        }, 2000);\n    }\n\n    initDb() {\n        //创建一个画板(需传入画板尺寸，将自动初始化)\n        this.db = new DrawingBoard(this.drawNode.width, this.drawNode.height);\n        //设置画板的绘图颜色（每次绘制前都可以重新设置）\n        this.lastLineWidth = 15;\n        this.db.setLineWidth(this.lastLineWidth);\n        // this.db.setColor(this.lastColor.r, this.lastColor.g, this.lastColor.b, this.lastColor.a);\n        //线条端点以圆角结尾\n        this.db.setLineCircleEnd(true);\n    }\n\n    initTexture() {\n        this.texture = new cc.RenderTexture();\n        this.texture.initWithSize(this.drawNode.width, this.drawNode.height, cc.RenderTexture.DepthStencilFormat.RB_FMT_S8);\n        let spf: cc.SpriteFrame = new cc.SpriteFrame(this.texture);\n        this.drawNode.getComponent(cc.Sprite).spriteFrame = spf;\n    }\n\n    initRead(){\n        this.targetCamera = this.node.getChildByName(\"tagCamera\").getComponent(cc.Camera);\n        var rander = new cc.RenderTexture();\n        rander.initWithSize(this.node.width, this.node.height, cc.RenderTexture.DepthStencilFormat.RB_FMT_S8);\n        this.targetCamera.targetTexture = rander;\n        this.targetCamera.render();\n        console.log(\"完成\");\n    }\n\n    onTouchStart(e: cc.Event.EventTouch) {\n        //将触摸位置作为线条的起点\n        //画板中使用的坐标系，与图片坐标系一样，原点在左上角，X轴向右为正，Y轴向下为正\n        //所以Y轴坐标应反过来, 这里用getLocationInView而不是getLocation\n   \n        this.touchId = e.getID();\n        if(this.touchId == 1){\n            this.touchScale = true;\n            return;\n        } \n\n        let pos = e.getLocation();\n        this.prePos = this.convertToDrawNodePos(pos);\n        this.startPos = this.convertToDrawNodePos(pos);\n        this.db.moveTo(this.prePos.x, this.prePos.y);\n        \n    }\n\n    onTouchMove(e: cc.Event.EventTouch) {\n        let touches = e.getTouches();\n        var touch1 = touches[0]\n        var delta1 = touch1.getDelta();\n        let pos = e.getLocation();\n \n        let pos1 = this.convertToDrawNodePos(touch1.getLocation());\n        let dst = this.startPos.sub(pos1).mag()\n        // this.label.string = touches.length + \"\";\n        if(touches.length == 1 && this.touchId < 1 && !this.touchScale && dst > 7){\n            // alert(\"不该进来\");\n            this.prePos = this.convertToDrawNodePos(pos);\n            var jg = this.pd(e);\n            this.changeColor(jg)\n            if (this.gameState == GameState.drawing) {\n                //从上一次绘制线条后的终点开始向鼠标当前位置绘制线条\n                this.db.lineTo(this.prePos.x, this.prePos.y);\n            } else if (this.gameState == GameState.erasing) {\n                // 橡皮擦\n                this.db.circle(this.prePos.x, this.prePos.y, 10);\n            }\n            //每次画板中的数据有变化后，及时将数据应用到贴图上，在屏幕上显示出来\n            this.drawToImg();\n        }\n        else if(touches.length == 2){\n            var touch1 = touches[0], touch2 = touches[1];\n            var delta1 = touch1.getDelta(), delta2 = touch2.getDelta();\n            var touchPoint1 = this.node.parent.convertToNodeSpaceAR(touch1.getLocation());\n            var touchPoint2 = this.node.parent.convertToNodeSpaceAR(touch2.getLocation());\n\n            var distance = touchPoint1.sub(touchPoint2);\n            var delta = delta1.sub(delta2);\n            var scale = 1;\n\n            if(Math.abs(distance.x) > Math.abs(distance.y)){\n                scale = (distance.x + delta.x) / distance.x * this.node.scale;\n            }\n            else{\n                scale = (distance.y + delta.y) / distance.y * this.node.scale;\n            }\n            if(scale > 2 ) scale = 2;\n            this.node.scale = scale <= 0.1 ? 0.1: scale;\n        }\n    }\n\n    onTouchEnd(e: cc.Event.EventTouch) {\n        this.touchId = e.getID();\n        if(this.touchId == 1) this.touchScale = false;\n        this.addHistory();\n    }\n\n    pd(event){\n        let cha = 2;\n        var pos = event.getLocation();\n        var jg = false;\n        for(let i=-cha; i<cha;i++){\n            let postion = cc.v2();\n            postion.x = pos.x + i;\n            for(let j =-cha; j<cha; j++){\n                postion.y = pos.y + j;\n                // console.log(\"检测点:\",postion.x,postion.y);\n                let img = this.getGraphisData(postion);\n                if((img[0] != 255 && img[1]!=255 && img[2]!=255)){\n                    jg = true;\n                    j = 10000; i = 10000;\n                }\n            }\n     \n        }\n        //\n        return jg;\n\n    }\n\n    convertToDrawNodePos(worldPos: cc.Vec2) {\n        let pos = this.drawNode.convertToNodeSpaceAR(worldPos);\n        pos.x += this.drawNode.width * this.drawNode.anchorX;\n        pos.y += this.drawNode.height * this.drawNode.anchorY;\n        pos.y = this.drawNode.height - pos.y;\n        return pos;\n    }\n\n    addHistory() {\n        let copy = this.db.copyData();\n        let ucopy = new Uint8Array(copy);\n        this.history.push({ data: ucopy });\n        // cc.log('历史步骤: ', this.history.length);\n    }\n\n    drawToImg() {\n        //获取画板中绘制的图案数据\n        let data: Uint8Array = this.db.getData();\n        //将数据传递给贴图对象\n        this.texture.initWithData(data, cc.Texture2D.PixelFormat.RGBA8888, this.db.width, this.db.height);\n    }\n\n    changeColor(red){\n        if(!red) this.db.setColor(this.errColor.r, this.errColor.g, this.errColor.b, this.errColor.a)\n        else this.db.setColor(this.lastColor.r, this.lastColor.g, this.lastColor.b, this.lastColor.a);\n    }\n\n    getGraphisData(point){\n        let Uint8 = new Uint8Array(4);\n        Uint8 = this.targetCamera.targetTexture.readPixels(Uint8,point.x,point.y,1,1);\n\n        return Uint8;\n    }\n\n\n    onBtnDraw() {\n        this.db.setLineWidth(this.lastLineWidth);\n        this.db.setColor(this.lastColor.r, this.lastColor.g, this.lastColor.b, this.lastColor.a);\n        this.gameState = GameState.drawing;\n    }\n\n    onBtnErase() {\n        this.db.setLineWidth(this.lastLineWidth * 3);\n        // 橡皮擦的颜色不能是(0,0,0,0),因为这样会和DrawingBoard里的默认颜色相同导致绘制跳过\n        this.db.setColor(255, 255, 255, 0);\n        this.gameState = GameState.erasing;\n\n    }\n\n    onBtnClear() {\n        this.db.reset();\n        this.drawToImg();\n        this.history.splice(0, this.history.length);\n    }\n\n    onBtnRevoke() {\n        this.history.pop();\n        if (this.history.length) {\n            let data: Uint8Array = this.history[this.history.length - 1].data;\n            this.db.setData(data.buffer);\n            this.texture.initWithData(this.db.getData(), cc.Texture2D.PixelFormat.RGBA8888, this.db.width, this.db.height);\n        } else {\n            this.onBtnClear();\n        }\n        cc.log('历史记录剩余: ', this.history.length);\n    }\n\n    onBtnSave() {\n        if (cc.sys.isBrowser) {\n            let width = this.drawNode.width;\n            let height = this.drawNode.height;\n\n            this.captureCamera.enabled = true;\n            let texture = new cc.RenderTexture();\n            texture.initWithSize(width, height, cc.RenderTexture.DepthStencilFormat.RB_FMT_S8);\n            this.captureCamera.targetTexture = texture;\n\n            let canvas: HTMLCanvasElement = document.createElement('canvas');\n            canvas.width = width;\n            canvas.height = height;\n            let ctx = canvas.getContext('2d');\n            this.captureCamera.render();\n            let data = texture.readPixels();\n            // write the render data\n            let rowBytes = width * 4;\n            for (let row = 0; row < height; row++) {\n                let srow = height - 1 - row;\n                let imageData = ctx.createImageData(width, 1);\n                let start = srow * width * 4;\n                for (let i = 0; i < rowBytes; i++) {\n                    imageData.data[i] = data[start + i];\n                }\n                ctx.putImageData(imageData, 0, row);\n            }\n            //\n            let dataUrl = canvas.toDataURL('image/png');\n            // cc.log('iamge-base64:', dataUrl);\n            let saveLink: any = document.createElementNS('http://www.w3.org/1999/xhtml', 'a');\n            saveLink.href = dataUrl;\n            saveLink.download = String(Date.now()) + '.png';\n            let event = document.createEvent('MouseEvents');\n            event.initMouseEvent('click', true, false, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null);\n            saveLink.dispatchEvent(event);\n            this.scheduleOnce(t => {\n                this.captureCamera.enabled = false;\n            }, 0.1);\n        } else {\n            cc.warn('暂时只支持web端保存图片');\n        }\n    }\n\n    update (dt) {\n        \n    }\n}\n"]}