html2canvas不支持文本缩放的解决方案
myzbx 2025-01-11 15:34 49 浏览
文章目录
- 背景
- 常规方案
- 出现问题
- html2canvas 原理解析
- 局限性
- 解决方案
- dom-to-image库
- dom - to - image 原理
- dom-to-image的相关方法
背景
前端同学在1k分辨率的屏幕上、开发8k分辨率的大屏(分辨率为8320 * 4320),为了能够在1k屏幕上正常显示,一般会根据真实屏幕大小进行动态计算,然后进行缩放(过程略过,大家可自行实现)以达到正常显示的效果。
现在有个弹框的内容,需要保存为图片,如下面的下载按钮,需要把内容下载为图片
常规方案
一般情况下,我们会使用html2canvas 来保存内容为图片。
安装依赖
yarn add html2canvas引入依赖
import html2canvas from "html2canvas";使用
给需要保存为图片的div区域加一个id名称downloadArea,然后在点击下载按钮时,执行以下方法:
// 下载
function handleDownload() {
  let el = document.getElementById("downloadArea");
  let options = {
    width: el.offsetWidth,
    height: el.offsetHeight,
    useCORS: true, // 是否尝试使用 CORS 从服务器加载图像
    allowTaint: false, // 是否允许跨源图像污染画布
    backgroundColor: "#121c2f", // 背景色
  }
  html2canvas(el, options).then((canvas) => {
    saveAs(canvas.toDataURL('image/jpeg'));
  });
}
function saveAs(dataUrl) {
  const link = document.createElement('a');
  link.href = dataUrl;
  link.download = `巡检统计_${new Date().getTime()}.jpg`;
  link.click();
  link.remove();
}
以上是常规的解决方案,对大多数情景都是很适用的。
出现问题
但是,当你兴冲冲地点击下载时,你看到导出的图片,文字大小没有缩放,如下所示:
html2canvas 原理解析
- 解析 DOM 结构html2canvas 首先会遍历目标 DOM 节点及其所有子节点。它会获取每个节点的位置(offsetLeft、offsetTop)、尺寸(offsetWidth、offsetHeight)、样式(包括width、height、background - color、border等各种 CSS 属性)等信息。例如,对于一个
- 元素,它会读取该元素的内联样式以及通过 CSS 类或 ID 应用的外部样式。
- 这个过程类似于浏览器渲染引擎在布局阶段对 DOM 树的处理。它会构建一个内部的数据结构来存储这些信息,为后续的绘制工作做准备。
- 创建画布(Canvas)在获取 DOM 信息后,html2canvas 会创建一个 HTML5 的Canvas元素。Canvas提供了一组用于在网页上绘制图形的 JavaScript API。例如,canvas.getContext(‘2d’)可以获取一个二维绘图上下文,用于绘制直线、矩形、圆形等各种图形。画布的大小通常会根据要捕获的 DOM 元素的大小来设置,以确保能够完整地容纳目标内容。
- 绘制 DOM 内容到画布对于文本内容,html2canvas 会使用Canvas的文本绘制函数,如fillText来将文本绘制到画布上。它会根据之前获取的字体样式(字体家族、字体大小、字体颜色等)来设置文本的样式。对于图像元素(<img>),它会获取图像的源(src)属性,通过drawImage函数将图像绘制到画布上。如果图像还没有加载完成,html2canvas 可能会等待图像加载后再进行绘制。对于其他元素,如背景颜色、边框等样式,也会通过相应的Canvas API 来实现。例如,fillStyle和strokeStyle用于设置填充颜色和边框颜色,fillRect和strokeRect用于绘制矩形的填充部分和边框。
- 将画布内容转换为图片数据当所有 DOM 内容都被正确地绘制到画布上后,html2canvas 会使用toDataURL方法将画布的内容转换为一个数据 URL。数据 URL 是一种可以直接在 HTML 中作为图像源(src)使用的格式,它以data:image/png;base64,开头,后面跟着经过 Base64 编码的图像数据。这个数据 URL 可以用于在新的<img>标签中显示,或者通过download属性让用户下载生成的图像。
局限性
html2canvas 在某些复杂的文本缩放场景下可能会出现问题。比如,它对一些 CSS3 的高级文本缩放特性支持可能有限。如果使用了transform: scale()等 CSS 属性来缩放文本和它的容器,html2canvas 可能不会完全按照浏览器渲染的效果进行捕获。
解决方案
小编盲猜一种思路(未尝试):在获取该id后:let el = document.getElementById("downloadArea");,给el加一个class样式,该样式仅在点击下载时才生效,然后写对应的样式。
简单的说就说:显示时是一种样式、下载时是另一种样式。
但是这样比较麻烦,那么有没有简单的方法呢?
这个问题问得好,有啊!
dom-to-image库
我们可以使用dom-to-image来做替代方案
安装依赖
yarn add dom-to-image引入dom-to-image
import domToImage from 'dom-to-image';使用
基本同上述的html2canvas写法
// 下载
function handleDownload() {
  const element = document.getElementById('downloadArea')
  element.style.backgroundColor = '#121c2f'
  const canvas = document.createElement('canvas')
  canvas.width = element.offsetWidth
  canvas.height = element.offsetHeight
  domToImage.toPng(element).then(function (canvas) {
    saveAs(canvas)
  })
}
function saveAs(dataUrl) {
  const link = document.createElement('a');
  link.href = dataUrl;
  link.download = `巡检统计_${new Date().getTime()}.jpg`;
  link.click();
  link.remove();
}这么做以后,我们看到下载的图片一切正常了,如下所示:
可以看到,分辨率为:2775 * 6280,也是8k分辨率下的弹框。
至此,完美解决。
dom - to - image 原理
创建虚拟 DOM 副本
- dom - to - image 首先会创建一个目标 DOM 元素的虚拟副本。这个副本与原始 DOM 在结构和样式上是相似的,但它是在内存中独立的一个表示。它会递归地遍历目标 DOM 树,复制每个节点的属性和样式信息。
- 例如,对于一个包含多个子元素的
- 元素,它会复制本身的样式(如width、height、background - color等),以及每个子元素(如、等)的样式和内容。
渲染虚拟 DOM 为 SVG 或其他格式
- 之后,dom - to - image 会将这个虚拟 DOM 渲染为 SVG(可缩放矢量图形)或其他矢量图形格式。SVG 是一种基于 XML 的矢量图形格式,它可以精确地描述图形的形状、颜色、位置等信息。
- 如果目标是保存为 PNG 等光栅图像格式,它会将 SVG 进一步转换。这个转换过程涉及到将矢量图形转换为像素点阵的形式,类似于在图形设计软件中导出图像的过程。
导出为图片数据
- 对于 SVG 格式,它可以直接作为一个独立的图形文件保存,或者通过一些工具将其转换为其他格式(如 PNG、JPEG 等)。如果是直接转换为 PNG 等格式,它会使用相关的图形处理库或浏览器的图形转换功能来完成转换。
- 最终,生成的图片数据可以像 html2canvas 一样,通过数据 URL 的形式提供给用户下载或在页面中显示。
dom-to-image的相关方法
附上dom-to-image的相关方法,以下参考前端将dom转换成图片_dom-to-image-CSDN博客的
相关推荐
- 如何设计一个优秀的电子商务产品详情页
- 
        加入人人都是产品经理【起点学院】产品经理实战训练营,BAT产品总监手把手带你学产品电子商务网站的产品详情页面无疑是设计师和开发人员关注的最重要的网页之一。产品详情页面是客户作出“加入购物车”决定的页面... 
- 怎么在JS中使用Ajax进行异步请求?
- 
        大家好,今天我来分享一项JavaScript的实战技巧,即如何在JS中使用Ajax进行异步请求,让你的网页速度瞬间提升。Ajax是一种在不刷新整个网页的情况下与服务器进行数据交互的技术,可以实现异步加... 
- 中小企业如何组建,管理团队_中小企业应当如何开展组织结构设计变革
- 
        前言写了太多关于产品的东西觉得应该换换口味.从码农到架构师,从前端到平面再到UI、UE,最后走向了产品这条不归路,其实以前一直再给你们讲.产品经理跟项目经理区别没有特别大,两个岗位之间有很... 
- 前端监控 SDK 开发分享_前端监控系统 开源
- 
        一、前言随着前端的发展和被重视,慢慢的行业内对于前端监控系统的重视程度也在增加。这里不对为什么需要监控再做解释。那我们先直接说说需求。对于中小型公司来说,可以直接使用三方的监控,比如自己搭建一套免费的... 
- Ajax 会被 fetch 取代吗?Axios 怎么办?
- 
        大家好,很高兴又见面了,我是"高级前端进阶",由我带着大家一起关注前端前沿、深入前端底层技术,大家一起进步,也欢迎大家关注、点赞、收藏、转发!今天给大家带来的主题是ajax、fetch... 
- 前端面试题《AJAX》_前端面试ajax考点汇总
- 
        1.什么是ajax?ajax作用是什么?AJAX=异步JavaScript和XML。AJAX是一种用于创建快速动态网页的技术。通过在后台与服务器进行少量数据交换,AJAX可以使网页实... 
- Ajax 详细介绍_ajax
- 
        1、ajax是什么?asynchronousjavascriptandxml:异步的javascript和xml。ajax是用来改善用户体验的一种技术,其本质是利用浏览器内置的一个特殊的... 
- 6款可替代dreamweaver的工具_替代powerdesigner的工具
- 
        dreamweaver对一个web前端工作者来说,再熟悉不过了,像我07年接触web前端开发就是用的dreamweaver,一直用到现在,身边的朋友有跟我推荐过各种更好用的可替代dreamweaver... 
- 我敢保证,全网没有再比这更详细的Java知识点总结了,送你啊
- 
        接下来你看到的将是全网最详细的Java知识点总结,全文分为三大部分:Java基础、Java框架、Java+云数据小编将为大家仔细讲解每大部分里面的详细知识点,别眨眼,从小白到大佬、零基础到精通,你绝... 
- 福斯《死侍》发布新剧照 "小贱贱"韦德被改造前造型曝光
- 
        时光网讯福斯出品的科幻片《死侍》今天发布新剧照,其中一张是较为罕见的死侍在被改造之前的剧照,其余两张剧照都是死侍在执行任务中的状态。据外媒推测,片方此时发布剧照,预计是为了给不久之后影片发布首款正式预... 
- 2021年超详细的java学习路线总结—纯干货分享
- 
        本文整理了java开发的学习路线和相关的学习资源,非常适合零基础入门java的同学,希望大家在学习的时候,能够节省时间。纯干货,良心推荐!第一阶段:Java基础重点知识点:数据类型、核心语法、面向对象... 
- 不用海淘,真黑五来到你身边:亚马逊15件热卖爆款推荐!
- 
        Fujifilm富士instaxMini8小黄人拍立得相机(黄色/蓝色)扫二维码进入购物页面黑五是入手一个轻巧可爱的拍立得相机的好时机,此款是mini8的小黄人特别版,除了颜色涂装成小黄人... 
- 2025 年 Python 爬虫四大前沿技术:从异步到 AI
- 
        作为互联网大厂的后端Python爬虫开发,你是否也曾遇到过这些痛点:面对海量目标URL,单线程爬虫爬取一周还没完成任务;动态渲染的SPA页面,requests库返回的全是空白代码;好不容易... 
- 最贱超级英雄《死侍》来了!_死侍超燃
- 
        死侍Deadpool(2016)导演:蒂姆·米勒编剧:略特·里斯/保罗·沃尼克主演:瑞恩·雷诺兹/莫蕾娜·巴卡林/吉娜·卡拉诺/艾德·斯克林/T·J·米勒类型:动作/... 
- 停止javascript的ajax请求,取消axios请求,取消reactfetch请求
- 
        一、Ajax原生里可以通过XMLHttpRequest对象上的abort方法来中断ajax。注意abort方法不能阻止向服务器发送请求,只能停止当前ajax请求。停止javascript的ajax请求... 
- 一周热门
- 最近发表
- 标签列表
- 
- HTML 简介 (30)
- HTML 响应式设计 (31)
- HTML URL 编码 (32)
- HTML Web 服务器 (31)
- HTML 表单属性 (32)
- HTML 音频 (31)
- HTML5 支持 (33)
- HTML API (36)
- HTML 总结 (32)
- HTML 全局属性 (32)
- HTML 事件 (31)
- HTML 画布 (32)
- HTTP 方法 (30)
- 键盘快捷键 (30)
- CSS 语法 (35)
- CSS 轮廓宽度 (31)
- CSS 谷歌字体 (33)
- CSS 链接 (31)
- CSS 定位 (31)
- CSS 图片库 (32)
- CSS 图像精灵 (31)
- SVG 文本 (32)
- 时钟启动 (33)
- HTML 游戏 (34)
- JS Loop For (32)
 
