docxtemplater避坑!!! 前端导出word怎么插入本地图片或base64 有完整示例

xgq11 2024-06-14 10:03:02 阅读 91

docxtemplater库实现前端通过模板导出word,遇到需求,要插图片并转成word并导出,在图片转换这块遇到了问题,网上查示例大多都跑不通,自己琢磨半天,总算搞明白了。

附上清晰完整示例,供参考。

如有不懂,私我询问!

首先需要一个word文件作为模板

test.docx

必须是docx文件!!!

{%} 代表图片 xgq是变量

安装需要的包

npm install docxtemplaternpm install docxtemplater-image-module-freenpm install pizzipnpm install file-savernpm install html2canvas # 如需截图的话 安装

import Docxtemplater from 'docxtemplater';import { saveAs } from 'file-saver';import PizZip from 'pizzip';import ImageModule from 'docxtemplater-image-module-free';import html2canvas from 'html2canvas';import image from './20240522152640.jpg';import docx from './test.docx';

插入本地图片并转换

const imageData = await fetch(image);const imageArrayBuffer = await imageData.arrayBuffer();const imgDataDict: Record<string, ArrayBuffer> = { xgq: imageArrayBuffer,};const docxData = await fetch(docx);const docxArrayBuffer = await docxData.arrayBuffer();const zip = new PizZip(docxArrayBuffer);const doc = new Docxtemplater(zip, { paragraphLoop: true, linebreaks: true, modules: [ new ImageModule({ getImage: (value: string, key: string) => { return imgDataDict[key]; }, getSize: (afterValue: ArrayBuffer, value: string, key: string) => { return [400, 400]; }, }), ],});doc.render({ xgq: "xgq", // 这里得是字符串否则会报错});const blob = doc.getZip().generate({ type: "blob", mimeType: "application/vnd.openxmlformats-officedocument.wordprocessingml.document",});saveAs(blob, "download.docx");

插入base64图片并转换

const base64 = "";const imageArrayBuffer = base64DataURLToArrayBuffer(base64);const imgDataDict: Record<string, ArrayBuffer> = { xgq: imageArrayBuffer,};const docxData = await fetch(docx);const docxArrayBuffer = await docxData.arrayBuffer();const zip = new PizZip(docxArrayBuffer);const doc = new Docxtemplater(zip, { paragraphLoop: true, linebreaks: true, modules: [ new ImageModule({ getImage: (value: string, key: string) => { return imgDataDict[key]; }, getSize: (afterValue: ArrayBuffer, value: string, key: string) => { return [400, 400]; }, }), ],});doc.render({ xgq: "xgq", // 这里得是字符串否则会报错});const blob = doc.getZip().generate({ type: "blob", mimeType: "application/vnd.openxmlformats-officedocument.wordprocessingml.document",});saveAs(blob, "download.docx");

const base64DataURLToArrayBuffer = (dataURL: string) => { const base64Regex = /^data:image\/(png|jpg|jpeg|svg|svg\+xml);base64,/; if (!base64Regex.test(dataURL)) { return false; } const stringBase64 = dataURL.replace(base64Regex, ""); let binaryString; if (typeof window !== "undefined") { binaryString = window.atob(stringBase64); } else { binaryString = new Buffer(stringBase64, "base64").toString("binary"); } const len = binaryString.length; const bytes = new Uint8Array(len); for (let i = 0; i < len; i++) { const ascii = binaryString.charCodeAt(i); bytes[i] = ascii; } return bytes.buffer;};

截图某个网页区域并插入转换

<div id="test" style={ { border: "1px solid red", width: 300 }}> <div>截图</div> <button type="button">666</button> <br /> <img src={ image} /></div>;

const dom: any = document.getElementById("test");const canvas = await html2canvas(dom, { useCORS: true, scale: 5,});const imageDataURL = canvas.toDataURL("image/png");const response = await fetch(imageDataURL);const imageArrayBuffer = await response.arrayBuffer();const imgDataDict: Record<string, ArrayBuffer> = { xgq: imageArrayBuffer,};const docxData = await fetch(docx);const docxArrayBuffer = await docxData.arrayBuffer();const zip = new PizZip(docxArrayBuffer);const doc = new Docxtemplater(zip, { paragraphLoop: true, linebreaks: true, modules: [ new ImageModule({ getImage: (value: string, key: string) => { return imgDataDict[key]; }, getSize: (afterValue: ArrayBuffer, value: string, key: string) => { return [400, 400]; }, }), ],});doc.render({ xgq: "xgq", // 这里得是字符串否则会报错});const blob = doc.getZip().generate({ type: "blob", mimeType: "application/vnd.openxmlformats-officedocument.wordprocessingml.document",});saveAs(blob, "download.docx");



声明

本文内容仅代表作者观点,或转载于其他网站,本站不以此文作为商业用途
如有涉及侵权,请联系本站进行删除
转载本站原创文章,请注明来源及作者。