用docxtemplater
库实现前端通过模板导出word,遇到需求,要插图片并转成word
并导出,在图片转换这块遇到了问题,网上查示例大多都跑不通,自己琢磨半天,总算搞明白了。
附上清晰完整示例,供参考。
如有不懂,私我询问!
首先需要一个word文件作为模板
必须是docx
文件!!!
{%}
代表图片 xgq
是变量
安装需要的包
npm install docxtemplater
npm install docxtemplater-image-module-free
npm install pizzip
npm install file-saver
npm 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");