百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

html2canvas不支持文本缩放的解决方案

zhezhongyun 2025-01-07 17:19 102 浏览

文章目录

  • 背景
  • 常规方案
  • 出现问题
    • 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博客

相关推荐

Python入门学习记录之一:变量_python怎么用变量

写这个,主要是对自己学习python知识的一个总结,也是加深自己的印象。变量(英文:variable),也叫标识符。在python中,变量的命名规则有以下三点:>变量名只能包含字母、数字和下划线...

python变量命名规则——来自小白的总结

python是一个动态编译类编程语言,所以程序在运行前不需要如C语言的先行编译动作,因此也只有在程序运行过程中才能发现程序的问题。基于此,python的变量就有一定的命名规范。python作为当前热门...

Python入门学习教程:第 2 章 变量与数据类型

2.1什么是变量?在编程中,变量就像一个存放数据的容器,它可以存储各种信息,并且这些信息可以被读取和修改。想象一下,变量就如同我们生活中的盒子,你可以把东西放进去,也可以随时拿出来看看,甚至可以换成...

绘制学术论文中的“三线表”具体指导

在科研过程中,大家用到最多的可能就是“三线表”。“三线表”,一般主要由三条横线构成,当然在变量名栏里也可以拆分单元格,出现更多的线。更重要的是,“三线表”也是一种数据记录规范,以“三线表”形式记录的数...

Python基础语法知识--变量和数据类型

学习Python中的变量和数据类型至关重要,因为它们构成了Python编程的基石。以下是帮助您了解Python中的变量和数据类型的分步指南:1.变量:变量在Python中用于存储数据值。它们充...

一文搞懂 Python 中的所有标点符号

反引号`无任何作用。传说Python3中它被移除是因为和单引号字符'太相似。波浪号~(按位取反符号)~被称为取反或补码运算符。它放在我们想要取反的对象前面。如果放在一个整数n...

Python变量类型和运算符_python中变量的含义

别再被小名词坑哭了:Python新手常犯的那些隐蔽错误,我用同事的真实bug拆给你看我记得有一次和同事张姐一起追查一个看似随机崩溃的脚本,最后发现罪魁祸首竟然是她把变量命名成了list。说实话...

从零开始:深入剖析 Spring Boot3 中配置文件的加载顺序

在当今的互联网软件开发领域,SpringBoot无疑是最为热门和广泛应用的框架之一。它以其强大的功能、便捷的开发体验,极大地提升了开发效率,成为众多开发者构建Web应用程序的首选。而在Spr...

Python中下划线 ‘_’ 的用法,你知道几种

Python中下划线()是一个有特殊含义和用途的符号,它可以用来表示以下几种情况:1在解释器中,下划线(_)表示上一个表达式的值,可以用来进行快速计算或测试。例如:>>>2+...

解锁Shell编程:变量_shell $变量

引言:开启Shell编程大门Shell作为用户与Linux内核之间的桥梁,为我们提供了强大的命令行交互方式。它不仅能执行简单的文件操作、进程管理,还能通过编写脚本实现复杂的自动化任务。无论是...

一文学会Python的变量命名规则!_python的变量命名有哪些要求

目录1.变量的命名原则3.内置函数尽量不要做变量4.删除变量和垃圾回收机制5.结语1.变量的命名原则①由英文字母、_(下划线)、或中文开头②变量名称只能由英文字母、数字、下画线或中文字所组成。③英文字...

更可靠的Rust-语法篇-区分语句/表达式,略览if/loop/while/for

src/main.rs://函数定义fnadd(a:i32,b:i32)->i32{a+b//末尾表达式}fnmain(){leta:i3...

C++第五课:变量的命名规则_c++中变量的命名规则

变量的命名不是想怎么起就怎么起的,而是有一套固定的规则的。具体规则:1.名字要合法:变量名必须是由字母、数字或下划线组成。例如:a,a1,a_1。2.开头不能是数字。例如:可以a1,但不能起1a。3....

Rust编程-核心篇-不安全编程_rust安全性

Unsafe的必要性Rust的所有权系统和类型系统为我们提供了强大的安全保障,但在某些情况下,我们需要突破这些限制来:与C代码交互实现底层系统编程优化性能关键代码实现某些编译器无法验证的安全操作Rus...

探秘 Python 内存管理:背后的神奇机制

在编程的世界里,内存管理就如同幕后的精密操控者,确保程序的高效运行。Python作为一种广泛使用的编程语言,其内存管理机制既巧妙又复杂,为开发者们提供了便利的同时,也展现了强大的底层控制能力。一、P...