基于Nodejs的word(office)文档实现方法(一)

    在线编辑office文档(word、excel、powerpoint等)的实现方式可以归集为两种:

    1、纯前端的在线编辑,如点聚WebOffice、金格WebOffice、PageOffice、ntko office、DSOFramer等,一般基于ActiveX控件实现,在各个浏览器逐步严控甚至禁止各类客户端控件的大背景下,这些解决方案的应用空间将会越来越小。当然近年也出现了一些基于云端office的解决方案,例如畅写office,前端不再依赖ActiveX控件,但云端服务器上的留痕始终让人不放心。

    2、后端创建、前端预览方式,如果要修改后台创建的office文档,则可以采用前端下载后修改,然后在上传更新后台文档;当然也可以有限集成纯前端在线编辑功能,以对后台创建的文档进行编辑,虽然笔者不建议这么做,毕竟前端的在线编辑功能再强大,也没有在自己电脑上用office进行编辑来的方便、灵活。

    本文主要介绍第二种方式,即“后端创建、前端预览”的相关实现方式,和注意事项。包括如下内容:

1、 开发环境

2、 后端创建方式一:新创建文档

3、 后端创建方式二:基于模板创建文档

4、 前端预览

5、 几个坑

一、 开发环境

    前端,vuejs,UI是quasar

    后端,nodejs

二、 后端创建方式一:新创建文档

    基于officegen(https://github.com/Ziv-Barber/officegen)实现office(word、excel、powerpoint)文档的创建。

   安装组件: npm i officegen --save-dev

    实现代码:

import fs from 'fs'

import path from 'path'

import officegen from 'officegen'

module.exports = {

    createpgbg:async (rwbh) => {

    //创建word对象

    let docx = officegen('docx') ;

    docx.on('finalize', function(written){

        console.log( '成功创建了word文件.' ) }) // Officegen calling this function to report errors:

    docx.on('error', function(err) { console.log(err) })

    let pObj = docx.createP() ;

    pObj.addText('Simple')

    pObj.addText(' with color', { color: '000088' })

    pObj.addText(' and back color.', { color: '00ffff', back: '000088' })

    pObj = docx.createP() pObj.addText('Since ')

    pObj.addText('officegen 0.2.12', { back: '00ffff', shdType: 'pct12', shdColor: 'ff0000' }) // Use pattern in the background.

    pObj.addText(' you can do ')

    pObj.addText('more cool ', { highlight: true }) // Highlight!

    pObj.addText('stuff!', { highlight: 'darkGreen' }) // Different highlight color.

    pObj = docx.createP()

    pObj.addText('Even add ')

    pObj.addText('external link', { link: 'https://github.com' })

    pObj.addText('!') pObj = docx.createP()

    pObj.addText('Bold + underline', { bold: true, underline: true })

    pObj = docx.createP({ align: 'center' })

    pObj.addText('Center this text', { border: 'dotted', borderSize: 12, borderColor: '88CCFF' })

    pObj = docx.createP()

    pObj.options.align = 'right'

    pObj.addText('Align this text to the right.')

    pObj = docx.createP()

    pObj.addText('Those two lines are in the same paragraph,')

    pObj.addLineBreak()

    pObj.addText('but they are separated by a line break.')

    docx.putPageBreak()

    pObj = docx.createP()

    pObj.addText('Fonts face only.', { font_face: 'Arial' })

    pObj.addText(' Fonts face and size.', { font_face: 'Arial', font_size: 40 })

    docx.putPageBreak()

    pObj = docx.createP()

    pObj.addImage('assets/uploads/1.jpg')

    let out = fs.createWriteStream('example1.docx')

    out.on('error', function(err) { console.log(err) })

    docx.generate(out)

}

}

以上代码生成的word效果


三、后端创建方式二:基于模板创建文档

    基于docxtemplater组件(https://github.com/open-xml-templating/docxtemplater)实现基于模板的office(word、excel、powerpoint)文档创建。

    安装组件:npm i docxtemplater --save-dev

                      npm i jszip@2.0.0 --save-dev

                       npm i docxtemplater-image-module-free--save-dev

    实现代码:

import fs from 'fs'

import path from 'path'

import jszip from 'jszip'

import docxtemplater from 'docxtemplater'

import docximagetemplater from'docxtemplater-image-module-free'

module.exports = {

  createpgbg:async(rwbh) => {

    //打开word模板文件

    let content =fs.readFileSync(path.resolve(__dirname,'../../assets/report/pgbgV2019.docx'),'binary');

    let zip = newjszip(content) ;

    let doc = newdocxtemplater() ;

    let opts = {

      centered:false,

     fileType:'docx',

      getImage:function(tagValue, tagName) {

        returnfs.readFileSync(path.join(__dirname, '../../assets/uploads/' + tagValue));

      },

      getSize:function(img, tagValue, tagName) {

        if(tagName== "a57"){

          return[350, 370];

        }

        elseif(tagName == "a58"){

          return[240, 180];

        }

        elseif(tagName == "a59"){

          return[240, 160];

        }

        else{

          return[100, 100];

        };

      }

    };

   doc.attachModule(new docximagetemplater(opts)) ;

   doc.loadZip(zip) ;

    //装载数据

    doc.setData({

      a53:'八一路',

      a54:'苏州东路',

      a55:'康平路',

      a56:'小区内水、电、气、有线电视等基础设施配套齐全,附近有学校、超市、菜场,商业、服务网点齐全,附近有公交站台,生活较方便',

     a57:'img1.jpg',

     a58:'img2.jpg',

     a59:'img3.jpg'

    }) ;

    try{

      //替换数据

      doc.render();

   }catch(error){};

    //保存文件

    let buf =doc.getZip().generate({type:'nodebuffer'}) ;

   fs.writeFileSync(path.resolve(__dirname,'../../assets/report/pgbg.docx'),buf);

    let filename ='2020-pgbg.docx' ;

    return{fileurl:filename} ;

  }

}


WORD模板


创建的word文档效果

四、前端预览

前端预览的选择挺多,其中有三种方式经过笔者实际使用,感觉挺不错:

方式1:<iframe

src='https://view.officeapps.live.com/op/view.aspx?src=你需要展示的word文档' />

方式2:<iframe src='http://ow365.cn/?i=1&furl=你需要展示的word文档' />

方式3:<iframe src='http://www.xdocin.com/xdoc?_func=to&_format=html&_cache=1&_xdoc=你需要展示的word文档' />


方式3的实现效果

     以上三种方式都要求“你需要展示的word文档”必须是互联网上能够访问的地址和文件名,此外一些使用心得如下:

    1、方式1免费,的文档地址必须用encodeURIComponent进行编码、地址必须为域名(不能用ip地址)、端口貌似也只能为80,文件大小有限制(word、ppt文件小于10M,excel文件小于5M)。

    2、方式2有限免费,使用前必须前在网站上注册(免费版单文件<5M、日访问量<500次)。但“你需要展示的word文档”支持IP地址访问,而且地址不需要编码。缺点就是每页都有一个OW365的LOGO,但不影响效果。

    3、方式3免费,而且支持IP地址访问、地址也不需要编码、不需要注册账号。缺点就是,显示的时候实际上是自动转换成了HTML,所以没有office里的那种显示质感。另外还有一个小缺点就是显示的速度有些慢,不过只要耐心,还是可以接受的。

五、几个坑

1、word模板中的图形显示模块

    Docxtemplater其实包含了images模块(即支持在word模板中插入图片的功能),但这个模块要收费,所以只能另找一个免费的images模块和Docxtemplater配合使用,而网上大部分文档都推荐使用open-docxtemplater-image-module,而且还提供了详细的实现代码,但经过笔者实际测试,不但显示不了图片,甚至还影响到Docxtemplater对文字模板的替换。

    而用docxtemplater-image-module-free组件,就可以正常配合Docxtemplater显示出图片,以上代码就是经过运行验证的。

2、压缩解压组件

    在word模板创建中需要用到压缩和解压组件,Docxtemplater只支持2.0版本的jszip,如果安装的是3.0+版本的jszip则在运行时就会报错,而且不能正确创建出word文档。这就是以上示例中强制安装2.0版jszip的原因。

    但如果使用pizzip,就没有版本的限制,可以正常使用。

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