前端导出word文档(基于vue)

工作中经常会遇到根据不同数据导出文档的情况,现在我们就来看一下基于 docxtemplater 来导出word文档的方法,使用起来非常便捷。

安装依赖的插件

-- 安装 docxtemplater
npm install docxtemplater pizzip  --save
-- 安装 jszip-utils
npm install jszip-utils --save 
-- 安装 jszip
npm install jszip --save
-- 安装 FileSaver
npm install file-saver --save

下面简单介绍一下这几个插件的功能:

  • docxtemplater:docxtemplate是一个从docx/pptx模板生成docx/pptx文档的库。语法包含变量替换、条件判断、循环、列表循环、表格循环等,还包括图片、html等模块转换。
  • jszip-utils:jszip-utils是一个与 JSZip 一起使用的跨浏览器实用程序集合,详细可参考 官网网址中的文档。
  • jszip: jszip是一个JavaScript类库,用来操作.zip文件的工具,详细可参考 官方网址 中的文档。
  • file-saver:file-saver是在客户端保存文件的解决方案,非常适合需要生成文件或者保存不应该发送到外部服务器的敏感信息的应用,详细用法可参考 官方网址 中的文档。

创建exportFile.js(导出word方法)

  1. 引入依赖包
  2. 使用 jszip-utils 读取并获得模板文件的二进制内容
  3. 使用 PizZip 创建实例,内容为模板的内容
  4. 使用 docxtemplater 创建实例,并加载 PizZip 创建的实例
  5. 使用docxtemplater 创建实例设置导出数据,并用导出数据的值替换所有模板变量
  6. 生成一个代表docxtemplater对象的zip文件
  7. 使用 saveAs 将生成的zip文件对象保存为目标类型的文件,并命名
import PizZip from 'jszip';
import docxtemplater from 'docxtemplater';
import JSZipUtils from 'jszip-utils';
import { saveAs } from 'file-saver';

/**
 * 导出word,支持图片
 * @param {Object} tempDocxPath 模板文件路径
 * @param {Object} wordData 导出数据
 * @param {Object} fileName 导出文件名
 */
export const exportWord = (tempDocxPath, wordData, fileName) => {
  // 读取并获得模板文件的二进制内容
  JSZipUtils.getBinaryContent(tempDocxPath, function (error, content) {
      if (error) {
        throw error;
      }

      // 创建一个PizZip实例,内容为模板的内容
      const zip = new PizZip(content);
      // 创建并加载docxtemplater实例对象
      const doc = new docxtemplater();
      doc.loadZip(zip);

      doc.setData(wordData);

      try {
        // 用模板变量的值替换所有模板变量
        doc.render();
      } catch (error) {
        // 抛出异常
        const e = {
          message: error.message,
          name: error.name,
          stack: error.stack,
          properties: error.properties
        };
        console.log(
          JSON.stringify({
            error: e
          })
        );
        throw error;
      }

      // 生成一个代表docxtemplater对象的zip文件(不是一个真实的文件,而是在内存中的表示)
      const out = doc.getZip().generate({
        type: 'blob',
        mimeType: 'application/vnd.openxmlformats-officedocument.wordprocessingml.document'
      });
      // 将目标文件对象保存为目标类型的文件,并命名
      saveAs(out, fileName);
    }
  );
};

编写word文档模板,具体的写法可以参考 docxtemplater demo 上的例子

下面列出几中常见的 docxtemplater 语法

  1. 替换
    js中的变量定义:
{title:'简介'}

word模板文件中语法:

{title}
  1. 循环
    js中的变量定义:
{
     loop:[
         { name: "Windows", price: 100},
         { name: "Mac OSX", price: 200 },
         { name: "Ubuntu", price: 0 }
    ],
    userGreeting: (scope) => {
        return "The product is" + scope.name + ", price:" + scope.price;
    },
}

word模板文件中语法:

循环{#loop} 
    {name}, {price}
     // 匿名函数插槽用法
     {userGreeting}
{/loop}
  1. 判断
    js中的变量定义:
{
    hasKitty: true,
    kitty: "Minie",
    hasDog: false,
    dog: "wangwang",
}

word模板文件中语法:

{#hasKitty}
    Cat’s name: {kitty}
{/hasKitty} 
{#hasDog}
    Dog’s name: {dog}
{/hasDog}
  1. 图片
    js中的变量定义:
{
    //文件路径
    image: '/logo.png',
}

word模板文件中语法:

{%image}

在组件中调用exportFile.js方法

import { exportWord } from '@/util/exportFile.js';

/**
* 参数1:模板文档路径
* 参数2:字段参数
* 参数3:输出文档
*/
const obj={
    ...
}
exportWord('template.docx' ,obj, '导出文件名.docx');

在这里关于模板文件的路径 tempDocxPath 我们要着重说一下。
在使用的过程中应该有不少人会遇到这种报错:Can't find end of central directory : is this a zip file ?

那这到底是什么原因导致的呢?
获得模板文件的二进制内容的方法,JSZipUtils.getBinaryContent(path, option) 提供path和option两个参数。我们来看一下path,前端开发最首先想到的可能是绝对路径或者相对路径,你可能还会用到@这个符号作为根目录使用,但在这里你显然不能这么用。

  1. 如果我们的模板文件为‘xxxx.docx’,那么我们的path应该为‘/xxxx.docx’。修改完这个地方之后,剩下的就是要知道你使用的vue-cli是版本2还是版本3。如果是2,则应该有一个static的文件夹,请见你的模板文件放入这个static文件夹中;如果是3,则有一个public文件夹,请将模板文件放入这个public文件夹中。
  2. 如果经过上面的修改依然报错,那么你就要看一下你的项目是不是微应用项目或者在项目路径上做了什么处理,如果是的话你需要根据项目目录填写xxxx.docx真正的路径。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,837评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,551评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,417评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,448评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,524评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,554评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,569评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,316评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,766评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,077评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,240评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,912评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,560评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,176评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,425评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,114评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,114评论 2 352

推荐阅读更多精彩内容

  • 在项目中,我们可以借助后端返回文件流实现文件下载。如果前端有数据,也可以借助前端框架进行下载。本文将介绍如何在前端...
    yichen_china阅读 3,430评论 0 3
  • 在项目中,我们可以借助后端返回文件流实现文件下载。如果前端有数据,也可以借助前端框架进行下载。本文将介绍如何在前端...
    Renaissance_阅读 28,584评论 23 25
  • 1 public下面新建word文档,填入Hi{nickName},nickName是字段名称,在需要导出word...
    明月清风_0e6e阅读 4,167评论 0 2
  • 操作步骤 1、安装插件 -- 安装 docxtemplatercnpm install docxtemplater...
    sincereXY阅读 2,380评论 0 1
  • 一、需要的js依赖 实现此功能需要使用到docxtemplater、jszip-utils、jszip、FileS...
    GL曲终人散阅读 40,477评论 47 16