前端html2canvas使用场景详解

html2canvas 是前端常用的 “DOM 转图片” 库,核心是将页面 DOM 节点渲染为 Canvas,再转为图片(Base64 或 Blob)。以下是 9 种核心使用场景的详细教程,包含代码示例、参数配置、问题解决,覆盖日常开发需求。

一、基础使用:将指定 DOM 转为 Base64 图片

适用于简单场景(如生成证书、截图分享),无需复杂配置。

1. 安装与引入

# npm 安装 npm install html2canvas --save 

javascript

// 模块化项目引入(Vue/React/Angular) import html2canvas from 'html2canvas'; // 非模块化项目(直接引入 CDN) <script src="https://cdn.jsdelivr.net/npm/[email protected]/dist/html2canvas.min.js"></script> 

2. 核心代码

html

<!-- 目标 DOM:需要转为图片的容器 --> <div> <h2>基础截图示例</h2> <p>这是要转为图片的内容</p> </div> <!-- 图片预览容器 --> <img /> 

javascript

// 触发截图(如点击按钮后) document.querySelector('#btn').addEventListener('click', async () => { // 1. 获取目标 DOM const targetDom = document.getElementById('targetDom'); // 2. 调用 html2canvas 生成 Canvas const canvas = await html2canvas(targetDom, { scale: 2, // 2倍缩放,避免图片模糊(关键参数) useCORS: true, // 允许跨域图片(如网络图片) logging: false, // 关闭控制台日志 backgroundColor: null // 保留 DOM 原有背景(默认白色) }); // 3. Canvas 转为 Base64 图片(支持 jpeg/png) const base64Img = canvas.toDataURL('image/jpeg', 1.0); // 1.0 表示质量(0-1) // 4. 预览图片 document.getElementById('previewImg').src = base64Img; // 5. (可选)下载图片 const link = document.createElement('a'); link.href = base64Img; link.download = '基础截图.jpg'; link.click(); }); 

二、进阶场景:处理跨域图片

当 DOM 中包含网络图片(如 https://xxx.com/xxx.jpg),易出现 “图片空白” 问题,需配置跨域参数。

关键配置

  1. 前端配置useCORS: true + allowTaint: true(允许跨域图片污染 Canvas);
  2. 后端配置:图片服务器需返回 Access-Control-Allow-Origin: *(或当前域名),否则跨域图片仍会空白。

完整代码

javascript

const canvas = await html2canvas(targetDom, { scale: 2, useCORS: true, // 允许加载跨域图片 allowTaint: true, // 允许 Canvas 被跨域图片污染(否则图片会被过滤) logging: false, // (可选)处理跨域图片加载失败的情况 onclone: (clonedDoc) => { // clonedDoc 是 DOM 克隆体,可在此修改图片地址 const imgs = clonedDoc.querySelectorAll('img'); imgs.forEach(img => { // 替换图片为同源地址(若后端未配置 CORS 时的备选方案) if (img.src.includes('跨域域名')) { img.src = img.src.replace('跨域域名', '同源代理域名'); } }); } }); 

三、复杂场景:固定宽高 + 滚动内容截图

适用于 “长列表截图”(如表格、聊天记录),需固定截图宽高,同时包含滚动区域的所有内容。

核心思路

  1. 先保存目标 DOM 的原始样式(overflow/height);
  2. 临时修改为 overflow: visible + 自适应高度(确保所有内容展开);
  3. 截图完成后恢复原始样式。

完整代码

javascript

async function captureScrollContent() { const targetDom = document.getElementById('targetDom'); // 1. 保存原始样式 const originalStyle = { overflow: targetDom.style.overflow, height: targetDom.style.height }; try { // 2. 临时修改样式:展开所有滚动内容 targetDom.style.overflow = 'visible'; targetDom.style.height = 'auto'; // 自适应高度,显示所有内容 // 3. 截图(固定宽高,避免内容变形) const canvas = await html2canvas(targetDom, { scale: 2, useCORS: true, logging: false, windowWidth: targetDom.offsetWidth, // 固定截图宽度 windowHeight: targetDom.offsetHeight // 固定截图高度(此时已包含所有内容) }); // 4. 转为图片并下载 const base64Img = canvas.toDataURL('image/png', 1.0); const link = document.createElement('a'); link.href = base64Img; link.download = '长列表截图.png'; link.click(); } finally { // 5. 恢复原始样式(无论成功失败都要恢复) targetDom.style.overflow = originalStyle.overflow; targetDom.style.height = originalStyle.height; } } // 调用 captureScrollContent(); 

四、Vue 项目:组件内 DOM 截图(含响应式数据)

Vue 中需注意 “DOM 渲染时机”—— 确保数据更新后(如接口请求完成)再截图,避免内容空白。

核心注意点

  1. 用 this.$nextTick() 确保 DOM 已更新;
  2. 若用 v-for 渲染列表,需等待列表渲染完成后再截图。

完整代码(Vue 3 示例)

vue

<template> <div> <!-- 目标 DOM:包含响应式数据 --> <div ref="certDom"> <h2>{{ userName }} 的证书</h2> <p>项目:{{ projectName }}</p> <p>时间:{{ date }}</p> </div> <button @click="generateCert">生成证书</button> <img :src="previewImg" alt="证书预览" /> </div> </template> <script setup> import { ref, nextTick } from 'vue'; import html2canvas from 'html2canvas'; // 响应式数据 const userName = ref(''); const projectName = ref(''); const date = ref('2025-01-01'); const previewImg = ref(''); const certDom = ref(null); // 目标 DOM 引用 // 接口请求:获取用户数据 const fetchUserInfo = async () => { const res = await fetch('/api/user'); const data = await res.json(); userName.value = data.userName; projectName.value = data.projectName; }; // 生成证书(截图) const generateCert = async () => { // 1. 先获取数据,确保 DOM 已更新 await fetchUserInfo(); // 2. 等待 Vue DOM 渲染完成 await nextTick(); // 3. 截图(certDom.value 是目标 DOM) const canvas = await html2canvas(certDom.value, { scale: 2, useCORS: true, backgroundColor: '#fff' }); // 4. 预览并下载 previewImg.value = canvas.toDataURL('image/jpeg', 1.0); const link = document.createElement('a'); link.href = previewImg.value; link.download = `${userName.value}的证书.jpg`; link.click(); }; </script> <style scoped> .cert-container { width: 500px; height: 700px; background: url('@/assets/cert-bg.jpg') no-repeat center; background-size: 100% 100%; padding: 50px; box-sizing: border-box; } </style> 

五、React 项目:函数组件截图(含 Hooks)

React 中需用 useRef 获取 DOM 节点,用 useEffect 处理异步渲染时机。

完整代码(React 18 示例)

jsx

import { useRef, useState } from 'react'; import html2canvas from 'html2canvas'; const ScreenshotComponent = () => { // 1. 获取目标 DOM 引用 const targetRef = useRef(null); // 2. 预览图片状态 const [previewImg, setPreviewImg] = useState(''); // 截图函数 const captureDom = async () => { if (!targetRef.current) return; // 3. 调用 html2canvas const canvas = await html2canvas(targetRef.current, { scale: 2, useCORS: true, logging: false }); // 4. 转为图片 const base64 = canvas.toDataURL('image/png', 1.0); setPreviewImg(base64); // 5. 下载图片 const link = document.createElement('a'); link.href = base64; link.download = 'react截图.png'; link.click(); }; return ( <div style={{ padding: 20 }}> {/* 目标 DOM:用 ref 绑定 */} <div ref={targetRef} style={{ width: 400, height: 300, background: '#e8f4f8', padding: 20 }} > <h3>React 截图示例</h3> <p>当前时间:{new Date().toLocaleString()}</p> </div> <button onClick={captureDom} style={{ margin: 20 }}>生成截图</button> {previewImg && <img src={previewImg} alt="预览" style={{ maxWidth: '100%' }} />} </div> ); }; export default ScreenshotComponent; 

六、特殊场景:隐藏 DOM 截图(不显示在页面)

适用于 “后台生成图片”(如生成后直接下载,无需在页面显示 DOM),需临时将 DOM 渲染到页面外。

核心思路

  1. 目标 DOM 初始样式设为 position: absolute; left: -9999px; top: -9999px;(隐藏在视口外);
  2. 截图完成后可删除该 DOM(可选)。

完整代码

html

<!-- 隐藏的目标 DOM:在视口外渲染 --> <div> <h2>隐藏 DOM 生成的图片</h2> <p>用户:张三</p> <p>等级:VIP</p> </div> <button>生成隐藏图片</button> 

javascript

document.getElementById('generateBtn').addEventListener('click', async () => { const hiddenDom = document.getElementById('hiddenDom'); // 截图(与正常 DOM 一致) const canvas = await html2canvas(hiddenDom, { scale: 2, useCORS: true, logging: false }); // 下载图片 const base64 = canvas.toDataURL('image/jpeg', 1.0); const link = document.createElement('a'); link.href = base64; link.download = '隐藏DOM图片.jpg'; link.click(); // (可选)截图后删除隐藏 DOM // hiddenDom.remove(); }); 

七、性能优化:批量截图 + 避免重复渲染

当需要批量生成图片(如多条数据生成证书),需控制截图顺序,避免同时渲染导致浏览器卡顿。

核心思路

  1. 用 async/await 按顺序截图(一条完成后再截下一条);
  2. 截图前判断是否已生成,避免重复操作。

完整代码

javascript

// 批量数据(如多条证书信息) const certList = [ { id: 1, userName: '张三', project: '前端开发' }, { id: 2, userName: '李四', project: '后端开发' }, { id: 3, userName: '王五', project: 'UI设计' } ]; // 批量截图函数 async function batchCapture() { for (const cert of certList) { // 1. 动态创建目标 DOM(每条数据一个 DOM) const tempDom = document.createElement('div'); tempDom.style.width = '500px'; tempDom.style.height = '700px'; tempDom.style.background = '#fff'; tempDom.style.padding = '50px'; tempDom.style.position = 'absolute'; tempDom.style.left = '-9999px'; // 隐藏在视口外 tempDom.innerHTML = ` <h2>${cert.userName} 的证书</h2> <p>项目:${cert.project}</p> <p>日期:2025-01-01</p> `; document.body.appendChild(tempDom); try { // 2. 截图(按顺序执行) const canvas = await html2canvas(tempDom, { scale: 2, useCORS: true, logging: false }); // 3. 下载图片 const base64 = canvas.toDataURL('image/jpeg', 1.0); const link = document.createElement('a'); link.href = base64; link.download = `${cert.userName}的证书.jpg`; link.click(); } finally { // 4. 移除临时 DOM(避免内存占用) document.body.removeChild(tempDom); // (可选)延迟 500ms,避免浏览器卡顿 await new Promise(resolve => setTimeout(resolve, 500)); } } } // 调用批量截图 batchCapture(); 

八、问题解决:常见报错与优化方案

1. 图片模糊

  • 原因:Canvas 默认缩放为 1,屏幕 DPI 过高(如 Retina 屏)导致模糊;
  • 解决:设置 scale: 2 或 scale: window.devicePixelRatio(自适应屏幕 DPI)。

javascript

const canvas = await html2canvas(targetDom, { scale: window.devicePixelRatio, // 自适应屏幕缩放 useCORS: true }); 

2. 跨域图片空白

  • 原因:图片服务器未配置 CORS,或未开启 useCORS: true
  • 解决
    1. 后端添加 Access-Control-Allow-Origin 响应头;
    2. 前端配置 useCORS: true + allowTaint: true
    3. 备选方案:用同源代理服务器转发图片(如 Nginx 代理)。

3. DOM 内容未完全渲染(如接口数据空白)

  • 原因:截图时机过早,DOM 尚未更新(如接口请求未完成);
  • 解决
    • Vue:用 this.$nextTick() 等待 DOM 渲染;
    • React:用 useEffect 或 await 等待数据加载;
    • 原生 JS:在接口 then 回调中执行截图。

4. 大 DOM 截图卡顿

  • 原因:DOM 节点过多(如长列表、复杂表格),渲染耗时;
  • 解决
    1. 简化 DOM(移除截图无关的节点,如隐藏按钮、多余注释);
    2. 分批截图(避免一次性渲染大量节点);
    3. 关闭日志:logging: false(减少控制台输出耗时)。

九、高级场景:Canvas 转 Blob 上传服务器

当需要将生成的图片上传到后端(而非本地下载),需将 Canvas 转为 Blob 格式(比 Base64 更节省带宽)。

完整代码

javascript

async function captureAndUpload() { const targetDom = document.getElementById('targetDom'); const canvas = await html2canvas(targetDom, { scale: 2, useCORS: true, logging: false }); // 1. Canvas 转为 Blob(支持指定格式和质量) canvas.toBlob(async (blob) => { // 2. 构建 FormData(用于上传) const formData = new FormData(); formData.append('image', blob, '上传图片.jpg'); // 第三个参数是文件名 formData.append('userId', '123'); // 其他参数(如用户ID) // 3. 上传到服务器 try { const res = await fetch('/api/upload-image', { method: 'POST', body: formData, // 注意:FormData 上传无需设置 Content-Type,浏览器会自动处理 }); const data = await res.json(); if (data.success) { alert('图片上传成功!'); console.log('图片地址:', data.imageUrl); } } catch (err) { console.error('上传失败:', err); } }, 'image/jpeg', 0.8); // 0.8 表示图片质量(0-1,值越高质量越好) } // 调用 captureAndUpload();

Read more

从GitHub镜像到本地运行:完整部署VoxCPM-1.5-TTS-WEB-UI语音模型教程

从GitHub镜像到本地运行:完整部署VoxCPM-1.5-TTS-WEB-UI语音模型教程 在智能音频内容爆发的今天,高质量、个性化的文本转语音(TTS)能力正成为越来越多应用的核心需求。无论是打造专属虚拟主播、构建离线语音助手,还是为视障用户提供无障碍阅读支持,一个能快速上手且音质出色的本地化TTS系统都显得尤为关键。 而开源社区中不断涌现的集成化项目,正在让这一切变得前所未有的简单。比如 VoxCPM-1.5-TTS-WEB-UI —— 它不是一个孤立的模型文件,而是一整套“开箱即用”的语音合成解决方案。你不需要逐行配置环境、手动安装依赖或调试CUDA版本兼容性,只需几步操作,就能在一个网页界面里输入文字,几秒后听到接近真人发音的44.1kHz高保真语音输出。 这背后到底发生了什么?它是如何把复杂的深度学习推理流程封装得如此简洁?更重要的是,我们该如何真正把它跑起来,并避免踩进那些看似不起眼却让人卡住半天的坑? VoxCPM-1.5-TTS-WEB-UI 是什么? 简单来说,这是一个将 VoxCPM-1.5 大规模语音生成模型 与 Web交互界面 深度整合的可执行镜像

基于DSOGI-PLL的C语言实现:STM32F407在不平衡电网中的高精度锁相与代码移植指南

1. 什么是DSOGI-PLL,为什么在不平衡电网中如此重要? 大家好,今天我想和大家聊聊一个在电力电子和嵌入式系统中非常实用的技术——基于双二阶广义积分器的锁相环(DSOGI-PLL)。如果你在做电网同步、逆变器控制或者新能源并网项目,尤其是在电网电压不平衡的情况下,DSOGI-PLL可能是你的救星。简单来说,DSOGI-PLL是一种高级的锁相技术,它能够在电网电压存在幅值不平衡、相位偏移甚至谐波干扰时,依然精准地追踪电网的相位和频率。 传统的锁相环在理想电网条件下表现良好,但现实中电网往往并不完美。比如,当三相电压幅值不一致,或者相位出现偏差时,普通锁相环可能会产生较大的误差,导致系统控制性能下降。而DSOGI-PLL通过双二阶广义积分器结构,能够有效分离正序和负序分量,从而在不平衡条件下实现高精度锁相。这对于STM32F407这样的嵌入式平台尤其重要,因为我们需要在资源有限的MCU上实现复杂的信号处理算法。 我在实际项目中多次用到DSOGI-PLL,尤其是在光伏逆变器和储能系统中。有一次,我们在一个偏远地区的微电网项目中遇到了严重的电压不平衡问题,普通锁相环根本无法稳定工作

从部署到实战:长亭雷池 WAF 全攻略(Web 安全防护高频场景落地指南)

从部署到实战:长亭雷池 WAF 全攻略(Web 安全防护高频场景落地指南)

从部署到实战:长亭雷池 WAF 全攻略(Web 安全防护高频场景落地指南) 长亭雷池(SafeLine)Web 应用防火墙作为国内顶尖安全厂商长亭科技推出的企业级 WAF 解决方案,凭借 “AI 智能防护、零误报率、易用性强” 的核心优势,成为政府、金融、电商、互联网等行业的首选 Web 安全防护产品。它不仅能精准防御 SQL 注入、XSS、命令执行等 OWASP Top 10 攻击,还支持 API 防护、爬虫治理、业务逻辑漏洞防护等高级功能,同时提供直观的可视化管理界面与完善的告警体系。本文将从部署准备、安装配置到高频场景实战,全方位拆解长亭雷池 WAF 的使用方法,帮助个人开发者、企业运维人员快速搭建专业级 Web 安全防护屏障。 一、长亭雷池

Go-Web从零开始的实战(六)

Go-Web从零开始的实战(六)

Cookie http是无状态的,Cookie是服务器“写给浏览器的纸条”。 无状态(Stateless)是http协议的核心设计,这意味着每次请求都是全新的、没有记忆的,即没有上下文。咋一看可能没感觉有啥问题,但是换个场景,你成功在网页上登录了你的账号,然后你想查看个人信息,但是服务器却不知道你是谁,因为你每次请求都是一个新的请求。 这可能听起来很荒唐,因为实际上我们大部分时候都很少遇到这种情况,因为服务器实际上通过多种方式(jwt自动续期、session等)避免了上述情况,Cookie也是一种方法。 我们再抽象下这个问题:http的无状态导致需要被记忆的信息无法被存储,那么解决方案最简单的就是每次请求都附带这些需要被记忆的信息就行。很明显这样确实可以解决问题,这里举个不恰当的例子:每次请求时都附带用户的用户名和密码,这样可以确保用户的信息,当然这很危险也很笨,你也不想每次都让服务器去数据库里面查询用户名和密码是否正确,但是这确实是一种办法。而且传输这些多余信息对网络带宽也是一种白白的占用。 Cookie的诞生 1994年,网景公司发明了Cookie,本质是服务器把“记