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

打印 上一主题 下一主题

主题 527|帖子 527|积分 1581

用docxtemplater库实现前端通过模板导出word,遇到需求,要插图片并转成word并导出,在图片转换这块遇到了题目,网上查示例大多都跑不通,本身琢磨半天,总算搞明确了。
附上清楚完备示例,供参考。
如有不懂,私我询问!

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


必须是docx文件!!!

{%} 代表图片 xgq是变量
安装需要的包

  1. npm install docxtemplater
  2. npm install docxtemplater-image-module-free
  3. npm install pizzip
  4. npm install file-saver
  5. npm install html2canvas # 如需截图的话 安装
复制代码
  1. import Docxtemplater from 'docxtemplater';
  2. import { saveAs } from 'file-saver';
  3. import PizZip from 'pizzip';
  4. import ImageModule from 'docxtemplater-image-module-free';
  5. import html2canvas from 'html2canvas';
  6. import image from './20240522152640.jpg';
  7. import docx from './test.docx';
复制代码
插入当地图片并转换

  1. const imageData = await fetch(image);
  2. const imageArrayBuffer = await imageData.arrayBuffer();
  3. const imgDataDict: Record<string, ArrayBuffer> = {
  4.   xgq: imageArrayBuffer,
  5. };
  6. const docxData = await fetch(docx);
  7. const docxArrayBuffer = await docxData.arrayBuffer();
  8. const zip = new PizZip(docxArrayBuffer);
  9. const doc = new Docxtemplater(zip, {
  10.   paragraphLoop: true,
  11.   linebreaks: true,
  12.   modules: [
  13.     new ImageModule({
  14.       getImage: (value: string, key: string) => {
  15.         return imgDataDict[key];
  16.       },
  17.       getSize: (afterValue: ArrayBuffer, value: string, key: string) => {
  18.         return [400, 400];
  19.       },
  20.     }),
  21.   ],
  22. });
  23. doc.render({
  24.   xgq: "xgq", // 这里得是字符串否则会报错
  25. });
  26. const blob = doc.getZip().generate({
  27.   type: "blob",
  28.   mimeType:
  29.     "application/vnd.openxmlformats-officedocument.wordprocessingml.document",
  30. });
  31. saveAs(blob, "download.docx");
复制代码
插入base64图片并转换

  1. const base64 =
  2.   "";
  3. const imageArrayBuffer = base64DataURLToArrayBuffer(base64);
  4. const imgDataDict: Record<string, ArrayBuffer> = {
  5.   xgq: imageArrayBuffer,
  6. };
  7. const docxData = await fetch(docx);
  8. const docxArrayBuffer = await docxData.arrayBuffer();
  9. const zip = new PizZip(docxArrayBuffer);
  10. const doc = new Docxtemplater(zip, {
  11.   paragraphLoop: true,
  12.   linebreaks: true,
  13.   modules: [
  14.     new ImageModule({
  15.       getImage: (value: string, key: string) => {
  16.         return imgDataDict[key];
  17.       },
  18.       getSize: (afterValue: ArrayBuffer, value: string, key: string) => {
  19.         return [400, 400];
  20.       },
  21.     }),
  22.   ],
  23. });
  24. doc.render({
  25.   xgq: "xgq", // 这里得是字符串否则会报错
  26. });
  27. const blob = doc.getZip().generate({
  28.   type: "blob",
  29.   mimeType:
  30.     "application/vnd.openxmlformats-officedocument.wordprocessingml.document",
  31. });
  32. saveAs(blob, "download.docx");
复制代码
  1. const base64DataURLToArrayBuffer = (dataURL: string) => {
  2.   const base64Regex = /^data:image\/(png|jpg|jpeg|svg|svg\+xml);base64,/;
  3.   if (!base64Regex.test(dataURL)) {
  4.     return false;
  5.   }
  6.   const stringBase64 = dataURL.replace(base64Regex, "");
  7.   let binaryString;
  8.   if (typeof window !== "undefined") {
  9.     binaryString = window.atob(stringBase64);
  10.   } else {
  11.     binaryString = new Buffer(stringBase64, "base64").toString("binary");
  12.   }
  13.   const len = binaryString.length;
  14.   const bytes = new Uint8Array(len);
  15.   for (let i = 0; i < len; i++) {
  16.     const ascii = binaryString.charCodeAt(i);
  17.     bytes[i] = ascii;
  18.   }
  19.   return bytes.buffer;
  20. };
复制代码
截图某个网页地区并插入转换

  1. <div id="test" style={{ border: "1px solid red", width: 300 }}>
  2.   <div>截图</div>
  3.   <button type="button">666</button>
  4.   <br />
  5.   <img src={image} />
  6. </div>;
复制代码
  1. const dom: any = document.getElementById("test");
  2. const canvas = await html2canvas(dom, {
  3.   useCORS: true,
  4.   scale: 5,
  5. });
  6. const imageDataURL = canvas.toDataURL("image/png");
  7. const response = await fetch(imageDataURL);
  8. const imageArrayBuffer = await response.arrayBuffer();
  9. const imgDataDict: Record<string, ArrayBuffer> = {
  10.   xgq: imageArrayBuffer,
  11. };
  12. const docxData = await fetch(docx);
  13. const docxArrayBuffer = await docxData.arrayBuffer();
  14. const zip = new PizZip(docxArrayBuffer);
  15. const doc = new Docxtemplater(zip, {
  16.   paragraphLoop: true,
  17.   linebreaks: true,
  18.   modules: [
  19.     new ImageModule({
  20.       getImage: (value: string, key: string) => {
  21.         return imgDataDict[key];
  22.       },
  23.       getSize: (afterValue: ArrayBuffer, value: string, key: string) => {
  24.         return [400, 400];
  25.       },
  26.     }),
  27.   ],
  28. });
  29. doc.render({
  30.   xgq: "xgq", // 这里得是字符串否则会报错
  31. });
  32. const blob = doc.getZip().generate({
  33.   type: "blob",
  34.   mimeType:
  35.     "application/vnd.openxmlformats-officedocument.wordprocessingml.document",
  36. });
  37. saveAs(blob, "download.docx");
复制代码
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

老婆出轨

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表