纯HTML无法生成可导出水印,必须依赖canvas或后端;CSS叠加仅限浏览器渲染层,导出时丢失;canvas通过加载源图、构造旋转透明水印纹理、逐像素绘制实现可靠水印。
纯 HTML 无法生成水印——它没有绘图能力、不支持图像合成,所谓“HTML 水印生成器”实际必须依赖 canvas 或后端服务。直接用 <div> 叠加文字只是视觉模拟,导出为图片时水印会丢失。
CSS 的 background-image 或伪元素 ::before 添加的文字/图案,仅在浏览器渲染层可见;一旦调用 html2canvas 或截图,若未显式绘制到 canvas 上,水印大概率不出现或错位。常见错误现象包括:导出 PNG 没水印、水印位置偏移、透明度失效、中文乱码。
canvas 是唯一能在前端可靠合成水印并导出图像的原生方案z-index 在跨 canvas 场景下无效canvas.toDataURL() 的跨域限制,会导致加载远程图片时抛出 SecurityError
关键不是“加文字”,而是“在目标图像像素上逐帧绘制水印纹理”。需分三步:加载源图 → 创建水印图案(含旋转、透明、重复)→ 绘制到 canvas 并导出。
drawImage() 先将原图完整绘制到 canvas 上createPattern() 构造带旋转的水印 canvas 作为纹理,避免逐次 fillText() 导致性能下降globalAlpha = 0.15 控制水印透明度,save()/restore() 保证状态隔离canvas.width/height 是否匹配原图尺寸,否则拉伸失真const watermarkCanvas = document.createElement('canvas');const ctx = watermarkCanvas.getContext('2d');watermarkCanvas.width = 200; watermarkCanvas.height = 200;ctx.rotate(-Math.PI / 6);ctx.font = 'bold 16px sans-serif';ctx.fillStyle = 'rgba(0,0,0,0.15)';ctx.textAlign = 'center';ctx.fillText('CONFIDENTIAL', 100, 100);
用户上传本地文件无跨域问题,但若允许粘贴 URL 或从图床加载图片,crossOrigin = "anonymous" 必须显式设置,否则 toDataURL() 报错;中文水印则需确保字体已加载或回退到系统默认字体。
立即学习“前端免费学习笔记(深入)”;
<img> 设置 crossOrigin="anonymous",且服务端需返回 Access-Control-Allow-Origin: *
ctx.font 中写死 "14px 'Microsoft YaHei', sans-serif" 更稳妥img.decode() 确保图片解码完成再绘制,防止 canvas 出现空白ctx.measureText().width 动态缩放字号,避免溢出真正可用的水印生成器,核心不在界面交互,而在 canvas 绘制逻辑是否绕过了跨域陷阱、是否把水印真正“烧录”进像素数据——这些细节漏掉一个,用户点下载就得到一张干净图。