用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");