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

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

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

文章目录

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

相关推荐

perl基础——循环控制_principle循环

在编程中,我们往往需要进行不同情况的判断,选择,重复操作。这些时候我们需要对简单语句来添加循环控制变量或者命令。if/unless我们需要在满足特定条件下再执行的语句,可以通过if/unle...

CHAPTER 2 The Antechamber of M de Treville 第二章 特雷维尔先生的前厅

CHAPTER1TheThreePresentsofD'ArtagnantheElderCHAPTER2TheAntechamber...

CHAPTER 5 The King&#39;S Musketeers and the Cardinal&#39;S Guards 第五章 国王的火枪手和红衣主教的卫士

CHAPTER3TheAudienceCHAPTER5TheKing'SMusketeersandtheCardinal'SGuard...

CHAPTER 3 The Audience 第三章 接见

CHAPTER3TheAudienceCHAPTER3TheAudience第三章接见M.DeTrévillewasatt...

别搞印象流!数据说明谁才是外线防守第一人!

来源:Reddit译者:@assholeeric编辑:伯伦WhoarethebestperimeterdefendersintheNBA?Here'sagraphofStea...

V-Day commemorations prove anti-China claims hollow

People'sLiberationArmyhonorguardstakepartinthemilitaryparademarkingthe80thanniversary...

EasyPoi使用_easypoi api

EasyPoi的主要特点:1.设计精巧,使用简单2.接口丰富,扩展简单3.默认值多,writelessdomore4.springmvc支持,web导出可以简单明了使用1.easypoi...

关于Oracle数据库12c 新特性总结_oracle数据库12514

概述今天主要简单介绍一下Oracle12c的一些新特性,仅供参考。参考:http://docs.oracle.com/database/121/NEWFT/chapter12102.htm#NEWFT...

【开发者成长】JAVA 线上故障排查完整套路!

线上故障主要会包括CPU、磁盘、内存以及网络问题,而大多数故障可能会包含不止一个层面的问题,所以进行排查时候尽量四个方面依次排查一遍。同时例如jstack、jmap等工具也是不囿于一个方面的问题...

使用 Python 向多个地址发送电子邮件

在本文中,我们将演示如何使用Python编程语言向使用不同电子邮件地址的不同收件人发送电子邮件。具体来说,我们将向许多不同的人发送电子邮件。使用Python向多个地址发送电子邮件Python...

提高工作效率的--Linux常用命令,能够决解95%以上的问题

点击上方关注,第一时间接受干货转发,点赞,收藏,不如一次关注评论区第一条注意查看回复:Linux命令获取linux常用命令大全pdf+Linux命令行大全pdf为什么要学习Linux命令?1、因为Li...

linux常用系统命令_linux操作系统常用命令

系统信息arch显示机器的处理器架构dmidecode-q显示硬件系统部件-(SMBIOS/DMI)hdparm-i/dev/hda罗列一个磁盘的架构特性hdparm-tT/dev/s...

小白入门必知必会-PostgreSQL-15.2源码编译安装

一PostgreSQL编译安装1.1下载源码包在PostgreSQL官方主页https://www.postgresql.org/ftp/source/下载区选择所需格式的源码包下载。cd/we...

Linux操作系统之常用命令_linux系统常用命令详解

Linux操作系统一、常用命令1.系统(1)系统信息arch显示机器的处理器架构uname-m显示机器的处理器架构uname-r显示正在使用的内核版本dmidecode-q显示硬件系...

linux网络命名空间简介_linux 网络相关命令

此篇会以例子的方式介绍下linux网络命名空间。此例中会创建两个networknamespace:nsa、nsb,一个网桥bridge0,nsa、nsb中添加网络设备veth,网络设备间...