cocos creator 小游戏区域截图功能实现
2019-08-14 10:17:55来源:博客园 阅读 ()
截图是游戏中非常常见的一个功能,在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); } });
原文链接:https://www.cnblogs.com/hutuzhu/p/11234136.html
如有疑问请与原作者联系
标签:
版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有
- cocos creator游戏开发 2019-08-14
- js原生开发一个2048小游戏 2019-05-16
- 用原生JS写一个网页版的2048小游戏(兼容移动端) 2019-03-10
- cocos 常用组件 2018-11-26
- HTML+Javascript制作拼图小游戏详解(二) 2018-11-02
IDC资讯: 主机资讯 注册资讯 托管资讯 vps资讯 网站建设
网站运营: 建站经验 策划盈利 搜索优化 网站推广 免费资源
网络编程: Asp.Net编程 Asp编程 Php编程 Xml编程 Access Mssql Mysql 其它
服务器技术: Web服务器 Ftp服务器 Mail服务器 Dns服务器 安全防护
软件技巧: 其它软件 Word Excel Powerpoint Ghost Vista QQ空间 QQ FlashGet 迅雷
网页制作: FrontPages Dreamweaver Javascript css photoshop fireworks Flash