前言
作为一个入职不久的前端小白来说,前段时间的项目中老大提了一个在我看来十分无理的需求,对!无理。他要求我用纯前端实现一个word文档导出功能,并且附模板。我当即就否决了他(这压根就触及到我知识盲区了…),随后他在长达两个半小时对我的思(hu)想(shuo)教(ba)育(dao)中。我当然是还是选择妥协了(毕竟自己老大我只能惯着啊…)。
准备工作
整个功能用到的相关技术还是比较多的,毕竟涉及到文件操作不是自己手撸封装的,自然得装不少轮子,其中最重要的是docxtemplater(重要归重要,这玩意儿得文档写的就是个坑)毕竟是个付费版,docxtemplater-image-module-free这个是个开源版本,他俩文档一样,但是少一个都不行。后来我用我仅有的两根脑神经理解了一下,后边这个是前边那个开源的一个图片导出的功能模块(毕竟前边那个是付费版…):
//-- 安装 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-image-module-free
npm install --save docxtemplater-image-module-free
代码实现
装好插件以后按照惯例开头引入组件页面,需要注意docxtemplater-image-module-free用Import引入可能会报错,需要换成require方式:
import docxtemplater from 'docxtemplater'
import PizZip from 'pizzip'
import JSZipUtils from 'jszip-utils'
import {
saveAs } from 'file-saver'
import ImageModule from 'docxtemplater-image-module-free' //这句是导出图片用的
/* 上边这行代码不好使的情况下需要用var ImageModule = require('docxtemplater-image-module-free') */
为了方便理解,生成word文档和生成图片分开写,首先写一个纯文本的word导出。写一个函数以便调用。
/* ******** base64DataURLToArrayBuffer函数代码块--导出图片函数编写位置 ******** */
function exportWord() {
// 设置模板变量的值,这个就是模板里所插入的内容数据,根据需要写成动态就好了。
var wordData = {
textcontents:'123456789',
//images: DataUrl //图片导出功能
};
// 读取并获得模板文件的二进制内容,是docxtemplater提供的固定写法
JSZipUtils.getBinaryContent("/exportTemplate.docx", function (error, content) {
// exportTemplate.docx是模板,React写在public里。我们在导出的时候,会根据此模板来导出对应的数据
//图片导出功能
/* var opts = {} opts.centered = false; //Set to true to always center images opts.fileType = "docx"; //Or pptx opts.getImage = function (chartId) { return base64DataURLToArrayBuffer(chartId); } //Pass the function that return image size opts.getSize = function () { return [450, 450]; } var imageModule = new ImageModule(opts); */
// 抛出异常
if (error) {
throw error;
}
// 创建一个PizZip实例,内容为模板的内容
let zip = new PizZip(content);
// 创建并加载docxtemplater实例对象
let doc = new docxtemplater()
//.attachModule(imageModule) //图片导出功能
.loadZip(zip)
.compile();
// 生成一个代表docxtemplater对象的zip文件(不是一个真实的文件,而是在内存中的表示)
doc.resolveData(wordData).then(function () {
console.log('ready');
doc.render();
var out = doc.getZip().generate({
type: "blob",
mimeType: "application/vnd.openxmlformats-officedocument.wordprocessingml.document",
});
// 将目标文件对象保存为目标类型的文件,并命名(起一个自己需要的名字就好了)
saveAs(out, "导出文件名.docx");
})
});
}
然后再随便撸个div简单写个按钮调用函数:
<div className="App">
<button type="info" onClick={
exportWord} >导出word文档</button>
</div>
到这里需要写一下word模板(光有代码没有容器是导不出来文件的/手动微笑),直接在{}中写入模板变量的键,导出后会自动插入值,文字大小,格式可以在word里设置:
导出后效果:
能成功导出文字模板这事儿就算成了,写下来就考虑怎么把图片搞进来(其实单图导出和多图的导出差不多,只要弄明白数据格式就可以。)
以下是两张base64的图片地址,图片的话需要处理成base64格式才能用,不然导出后图片可能会不显示:
const DataUrl = [
{
src: ''
},
{
src: ''
}
]
有了base64的图片地址以后,需要调用docxtemplater提供的base64DataURLToArrayBuffer
方法把地址处理成ArrayBuffer才能正常导出图片,否自会出现报错。(函数写的位置在之前代码中注释出来了)
function base64DataURLToArrayBuffer(dataURL) {
console.log(dataURL);
const base64Regex = /^data:image\/(png|jpg|svg|jpeg|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;
}
做完这些以后,把之前纯文本导出的代码中带*//图片导出功能
*的注释取消注释功能就实现了,最后记得把word模板做如下修改(有了代码还得有容器才能导出word呢):
结语
回头写这篇博客的时候,才发现这个功能实现起来是真的简单…
参考
React纯前端生成word文档(支持多图片导出)
React纯前端生成word文档(支持多图片导出)
https://docxtemplater.readthedocs.io/en/latest/index.html
今天的文章react前端开发教程_react引入图片分享到此就结束了,感谢您的阅读。
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
如需转载请保留出处:https://bianchenghao.cn/82076.html