前端如何实现一键截图功能?

开发 前端
网页截图功能目前也是非常常见的需求, 尤其是在在线教育领域. 我们朋友圈的微信海报, 活动海报等, 一般都是运营/市场人员通过设计工具设计而成, 但是如何更好的映射到自己的服务体系里面, 比如H5页面中, 植入更多信息收集, 交互能力. 这一块的应用探索, 页面截图是一个非常好的解决方案.

[[353862]]

 前言

网页截图功能目前也是非常常见的需求, 尤其是在在线教育领域. 我们朋友圈的微信海报, 活动海报等, 一般都是运营/市场人员通过设计工具设计而成, 但是如何更好的映射到自己的服务体系里面, 比如H5页面中, 植入更多信息收集, 交互能力. 这一块的应用探索, 页面截图是一个非常好的解决方案.

接下来笔者就来复盘一下如何基于网页, 一键生成页面海报的功能, 并将此能力, 集成到笔者的开源项目H5-Dooring中为编辑器赋能.

正文

在实现具体功能之前, 我们先看看具体的实现效果:


从演示中我们可以看出, 我们最终目标是实现在PC端生成H5页面的截图, 所以可能会涉及到以下几个问题:

  • 如何实现将页面转化为图片
  • 如何实现H5效果模拟并截取实际的H5页面

我们可以先来想想实现思路, 如何能基于dom转化为图片? 这块技术也是老生常谈的课题了, 我们都知道可以用canvas来实现, 大致流程如下: 


我们如果用原生的实现方案, 大致要经历以上几个步骤, 其中第二步是关键环节也是最复杂的一步, 我们需要手动实现dom到canvas的映射, 最后转化为标准的canvas绘图对象. 当然现成也有很多库可以直接帮我们简化这一步骤, 比如html2canvas, dom-to-image. 接下来我们就来解决第一个问题.

如何实现将页面转化为图片

在亲自调研了html2canvas库并使用的过程中, 笔者发现了很多问题, 比如如果样式中出现%单位, 或者有一些图片背景的问题, 导致html2canvas并没有很好的work, 而且渲染还原度和清晰度都有问题, 所以笔者暂时没有深入研究(不过这些问题可以通过修改库本身解决), 后面笔者直接用了dom-to-image, 发现使用起来很简单, 而且几乎不会出现上面说的这些问题, 所以笔者果断采用了dom-to-image, 后面看了该库的源码, 感觉写的也很优雅易懂, 后期做二次开发应该问题不是很大. 我们可以看看其官网的基本使用:

  1. // 引入 
  2. import domtoimage from 'dom-to-image'
  3.  
  4. // 生成图片 
  5. domtoimage.toPng(node) 
  6.     .then(function (dataUrl) { 
  7.         var img = new Image(); 
  8.         img.src = dataUrl; 
  9.         document.body.appendChild(img); 
  10.     }) 
  11.     .catch(function (error) { 
  12.         console.error('oops, something went wrong!', error); 
  13.     }); 

用法也很简单, 而且它提供了足够多的配置项, 我们可以灵活配置.


第一个问题就这么解决了, 不过在使用过程中发现图片模糊的问题, 这块网上也有很多解决方案. 比如先放大dom, 在处理成canvas最后生成图片的时候在缩小等, 这块笔者就不一一举例了.

如何实现H5效果模拟并截取实际的H5页面

因为我们设计的H5页面都在pc端完成的, 所以要想生成H5预览图, 无非是本地模拟尺寸, 进行渲染, 具体方案如下:

  • 采用iframe作为H5页面容器去生成截图
  • 直接限制宽度在当前页面生成截图
  • 采用服务端爬虫一键模拟手机访问生成截图

上面说的方案都可以尝试, 第三种方案笔者之前也开源过爬虫应用来解决这个问题, 感兴趣的可以研究了解一下, 我们很明显会选择第一种方案来实现, 就如演示中的, 我们看到的弹窗中的H5其实是在iframe中渲染的:

实现思路有了, 该问题也就很好实现了, 我们只需要在父页面和iframe实现消息通信即可, 比如在iframe加载完成之后手动通知iframe截取自身. 基本实现代码如下:

  1. // 编辑器页面, 也就是父页面 
  2. // 定义截图子页面句柄函数 
  3. window.getFaceUrl = (url) => { 
  4.   setFaceUrl(url) 
  5.   setShowModalIframe(false
  6.  
  7. // iframe页面, 也就是预览页面 
  8. const generateImg = (cb:any) => { 
  9.     domtoimage.toBlob(refImgDom.current,  
  10.       { 
  11.         width, 
  12.         height, 
  13.       } 
  14.     ) 
  15.     .then(function (blob:Blob) { 
  16.         const formData = new FormData(); 
  17.         formData.append('file', blob, 'tpl.jpg'); 
  18.         req.post('/files/upload/free', formData).then((res:any) => { 
  19.           cb && cb(res.url) 
  20.         }) 
  21.     }) 
  22.     .catch(function (error:any) { 
  23.         console.error('oops, something went wrong!', error); 
  24.     }); 
  25.  
  26. // 触发父页面的方法,将图片传给父页面 
  27. generateImg((url:string) => { 
  28.   parent.window.getFaceUrl(url); 
  29. }) 

 

责任编辑:姜华 来源: 趣谈前端
相关推荐

2023-12-22 13:57:00

文本复制长按复制功能JS

2011-09-15 19:05:49

windows 7一键关机

2022-04-07 13:56:13

前端一键换肤

2021-02-22 11:07:49

Windows 10微软数据传输

2015-02-09 15:25:52

换肤

2012-03-01 14:00:08

2018-03-20 12:30:44

iOSPush抓包

2023-11-02 10:14:50

TinykeysWeb应用

2023-05-05 17:11:11

前端Mousetrap方式

2021-08-03 15:25:09

数据库Sharding SpSQL

2022-11-24 16:11:27

2022-11-23 10:22:12

组件库前端

2011-07-03 16:05:50

2021-10-14 08:40:58

前端技术数据可视化

2018-08-29 08:19:43

iPhone求救手机

2022-07-22 14:32:29

账号登录服务鸿蒙

2021-10-19 05:49:15

Windows 11操作系统微软

2009-07-07 08:44:52

微软Windows 7新功能

2022-12-01 17:46:53

网页变灰功能前端

2015-11-03 15:29:49

ONOS开放网络操作系统SDN
点赞
收藏

51CTO技术栈公众号