zoukankan      html  css  js  c++  java
  • cocos creator 小游戏区域截图功能实现

    截图是游戏中非常常见的一个功能,在cocos中可以通过摄像机和 RenderTexture 可以快速实现一个截图功能,具体API可参考:https://docs.cocos.com/creator/manual/zh/render/camera.html?h=%E6%88%AA%E5%9B%BE,其中官方也提供了比较完整的例子。

    实际上不用官网提供的全屏截图的例子,一般在网页中我们也能将页面截图保存,比如通过htmltocanvas,cocos开发的小游戏在网页中打开实际就是一个canvas,前端是可以通过将canvas保存为图片的,这里就不细说了。

    我们还是来看下如何把屏幕中某一区域的内容生成图片并保存到本地。

    1、创建RenderTexture

    //新建一个 RenderTexture,并且设置 camera 的 targetTexture 为新建的 RenderTexture,这样 camera 的内容将会渲染到新建的 RenderTexture 中。
    let texture = new cc.RenderTexture();
    let gl = cc.game._renderContext;
    //如果截图中不含mask组件可以不加第三个参数,不过建议加上 texture.initWithSize(this.node.width, this.node.height, gl.STENCIL_INDEX8);//这里的宽高直接决定了截图的宽高,如果是全屏截图就是cc.visibleRect.width, cc.visibleRect.height,该处可以设置为截图目标区域的宽高
    this.camera = this.node.addComponent(cc.Camera); this.camera.targetTexture = texture; this.texture = texture;

     2、绘制canvas

    createSprite() {
            let width = this.texture.width;
            let height = this.texture.height;
         //截图的本质是创建一个canvas,然后通过canvas生成图片材质
    if (!this._canvas) { this._canvas = document.createElement('canvas'); this._canvas.width = width; this._canvas.height = height; } else { this.clearCanvas(); } let ctx = this._canvas.getContext('2d'); this.camera.render();//相机绘制,将屏幕上的内容更新到renderTexture中 let data = this.texture.readPixels();//读取renderTexture中的数据 let rowBytes = width * 4; for (let row = 0; row < height; row++) { let srow = height - 1 - row; let imageData = ctx.createImageData(width, 1); let start = srow * width * 4; for (let i = 0; i < rowBytes; i++) { imageData.data[i] = data[start + i]; } ctx.putImageData(imageData, 0, row); } return this._canvas; },

    上述代码中用到了canvas 的createImageData() 和putImageData()方法,createImageData() 方法创建新的空白 ImageData 对象,putImageData() 方法将图像数据(从指定的 ImageData 对象)放回画布上。

    3、获取图片

    initImage(img) {
            // return the type and dataUrl
            var dataURL = this._canvas.toDataURL("image/png");
            var img = document.createElement("img");
            img.src = dataURL;
            return img;
    },

    生成canvas就可以通过canvas.toDataURL()方法将canvas转换为图片

    4、生成截图效果,将上一步生成的图片当做材质挂载到新建的node

    showSprite(img) {
            let y = this.getTargetArea().y;
            let x = this.getTargetArea().x;
            let rect = new cc.Rect(x, y, 770, 800)
            let texture = new cc.Texture2D();
            texture.initWithElement(img);
    
            let spriteFrame = new cc.SpriteFrame();
            spriteFrame.setTexture(texture);
            spriteFrame.setRect(rect)
    
            let node = new cc.Node();
            let sprite = node.addComponent(cc.Sprite);
            sprite.spriteFrame = spriteFrame;
    
            node.zIndex = cc.macro.MAX_ZINDEX;
            node.parent = cc.director.getScene();
            // set position
            let width = cc.winSize.width;
            let height = cc.winSize.height;
            node.x = width / 2;
            node.y = height / 2;
            node.on(cc.Node.EventType.TOUCH_START, () => {
                node.parent = null;
                node.destroy();
            });
            this.captureAction(node, width, height);
        },
    

    5、截图动画(类似手机截图,截图后有个缩略图动画)

    captureAction(capture, width, height) {
            let scaleAction = cc.scaleTo(1, 0.3);
            let targetPos = cc.v2(width - width / 6, height / 4);
            let moveAction = cc.moveTo(1, targetPos);
            let spawn = cc.spawn(scaleAction, moveAction);
    
            let finished = cc.callFunc(() => {
                capture.destroy();
            })
            let action = cc.sequence(spawn, finished);
            capture.runAction(action);
        },
    

    6、下载图片到本地,动态生成a标签,模拟点击后移除

    downloadImg() {
            this.createSprite();
            var img = this.initImage();
            this.showSprite(img)
            var dataURL = this._canvas.toDataURL("image/png")
            var a = document.createElement("a")
            a.href = dataURL;
            a.download = "image";
            document.body.appendChild(a);
            a.click();
            document.body.removeChild(a);
        },

     完整代码如下:

    cc.Class({
        extends: cc.Component,
    
        properties: {
            _canvas: null,
            targetNode: cc.Node
        },
        onLoad() {
            this.init();
        },
    
        init() {
            let texture = new cc.RenderTexture();
            let gl = cc.game._renderContext;
            texture.initWithSize(this.node.width, this.node.height, gl.STENCIL_INDEX8);
            this.camera = this.node.addComponent(cc.Camera);
            this.camera.targetTexture = texture;
            this.texture = texture;
        },
        // create the img element
        initImage(img) {
            // return the type and dataUrl
            var dataURL = this._canvas.toDataURL("image/png");
            var img = document.createElement("img");
            img.src = dataURL;
            return img;
        },
        // create the canvas and context, filpY the image Data
        createSprite() {
            let width = this.texture.width;
            let height = this.texture.height;
            if (!this._canvas) {
                this._canvas = document.createElement('canvas');
                this._canvas.width = width;
                this._canvas.height = height;
            } else {
                this.clearCanvas();
            }
            let ctx = this._canvas.getContext('2d');
            this.camera.render();
            let data = this.texture.readPixels();
            // write the render data
            let rowBytes = width * 4;
            for (let row = 0; row < height; row++) {
                let srow = height - 1 - row;
                let imageData = ctx.createImageData(width, 1);
                let start = srow * width * 4;
                for (let i = 0; i < rowBytes; i++) {
                    imageData.data[i] = data[start + i];
                }
    
                ctx.putImageData(imageData, 0, row);
            }
            return this._canvas;
        },
        getTargetArea() {
            let targetPos = this.targetNode.convertToWorldSpaceAR(cc.v2(0, 0))
            let y = cc.winSize.height - targetPos.y - this.targetNode.height / 2;
            let x = cc.winSize.width - targetPos.x - this.targetNode.width / 2;
            return {
                x,
                y
            }
        },
        downloadImg() {
            this.createSprite();
            var img = this.initImage();
            this.showSprite(img)
            var dataURL = this._canvas.toDataURL("image/png")
            var a = document.createElement("a")
            a.href = dataURL;
            a.download = "image";
            document.body.appendChild(a);
            a.click();
            document.body.removeChild(a);
        },
        // show on the canvas
        showSprite(img) {
            let y = this.getTargetArea().y;
            let x = this.getTargetArea().x;
            let rect = new cc.Rect(x, y, 770, 800)
            let texture = new cc.Texture2D();
            texture.initWithElement(img);
    
            let spriteFrame = new cc.SpriteFrame();
            spriteFrame.setTexture(texture);
            spriteFrame.setRect(rect)
    
            let node = new cc.Node();
            let sprite = node.addComponent(cc.Sprite);
            sprite.spriteFrame = spriteFrame;
    
            node.zIndex = cc.macro.MAX_ZINDEX;
            node.parent = cc.director.getScene();
            // set position
            let width = cc.winSize.width;
            let height = cc.winSize.height;
            node.x = width / 2;
            node.y = height / 2;
            node.on(cc.Node.EventType.TOUCH_START, () => {
                node.parent = null;
                node.destroy();
            });
            this.captureAction(node, width, height);
        },
        // sprite action
        captureAction(capture, width, height) {
            let scaleAction = cc.scaleTo(1, 0.3);
            let targetPos = cc.v2(width - width / 6, height / 4);
            let moveAction = cc.moveTo(1, targetPos);
            let spawn = cc.spawn(scaleAction, moveAction);
    
            let finished = cc.callFunc(() => {
                capture.destroy();
            })
            let action = cc.sequence(spawn, finished);
            capture.runAction(action);
        },
    
        clearCanvas() {
            let ctx = this._canvas.getContext('2d');
            ctx.clearRect(0, 0, this._canvas.width, this._canvas.height);
        }
    });
    

    网络释义
    RenderTexture: 渲染纹理
  • 相关阅读:
    vue 监听变量或对象
    同行span标签设置display:inline-block;overflow:hidden垂直对齐问题
    vue style 的scoped 使用
    判断两个对象是否相等
    js call 和 apply方法记录
    接口联调遇到的坑 总结
    onclick或者其他事件在部分移动端无效的问题
    js 迭代方法
    原生js实现类的添加和删除,以及对数据的add和update、view ,ajax请求 ,页面离开的操作,获取url中参数等方法
    input 的radio checkbox 和 select 相关操作
  • 原文地址:https://www.cnblogs.com/hutuzhu/p/11234136.html
Copyright © 2011-2022 走看看