目录
- 需求
- 实现方案
- puppeteer
- node-canvas
- 浏览器
- 问题分解
- 具体实现
- 启动一个本地服务器
- 创建html,引入资源库
- 解析xls文件
- 写入中间logo
- 写入底部文字
- canvas转化为图片,并下载到本地
- 递归调用
- 最终效果
需求
一次普通的技术需求会议
项目经理首先发言 我们技术这边需要将xls表格中的几千条数据变成二维码,并且中间镶嵌logo,图片底部放置编号,由于xls表格数据私密,不能通过第三方完成
平常这个事情都是后端处理的,前端就是来摸鱼的,但是这次一反常态,后端脸黑了,带样式搞不来,脚一蹬,直接装死
项目经理用期盼的眼神看着我,顿时我紧张了起来,眼神飘忽,我已经好多年没搞过node了啊!!会议室都沉默了,在项目经理不断精神攻击下,后端装死的情况下,看来注定要大前端来拯救世界了,毕竟JavaScript万能语言,俺来试试吧!
实现方案
以上情节纯属虚构,但是需求确实是这样的,虽然好几年没碰过node,好歹年轻记性好,用过的基本都还记得,调研实现方案上没出现太多问题,有如下方案
puppeteer
地址
使用基于node环境的puppeteer,进行二维码绘制,图片绘制,是JavaScript开发者面对此类需求的主流选择
node-canvas
地址
同样是在服务端完成渲染,但是这个库依赖node-gyp,如果不安装python2,那安装过程懂得都懂,不过这也是很不错的方案
浏览器
通过浏览器canvas绘制,然后下载下来,会有刷刷刷下载图片的炫酷效果
很明显有刷刷刷下载图片炫酷效果的方案更好,所以就选择你了 浏览器方案!
问题分解
确定了技术方案,就要考虑具体实现了
- JavaScript读取execl文件,并处理成理想格式
- 将读取到的execl中的网址字段生成一张二维码
- 将二维码写入canvas,在其中间加上logo,并在底部加一行文字
- 将canva转化为DataURL,下载它
- 不断递归生成,直到xls数据全部处理完毕
理论存在,实践开始!
具体实现
启动一个本地服务器
首先我们通过VScode Live Server 启动一个本地服务器
这里有好奇宝宝要问了,为啥第一步是这?
答:因为浏览器是访问不了电脑的文件系统的,所以只能通过启动一个本地服务器的方案,来读取我们的资源文件
创建html,引入资源库
分析需要用到的第三方开源库
解析xls
生成QRcode
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8" /> <meta http-equiv="X-UA-Compatible" content="IE=edge" /> <meta name="viewport" content="width=device-width, initial-scale=1.0" /> <title>生成二维码</title> <script src="./qrcode.js"></script> <script src="./xlsx.full.min.js"></script> </head> <body> <!-- 用于生成载体 最终生成的图片大小,按自己的需求来 --> <canvas width="260" height="310" id="canvas"></canvas> </body> <script> const ctx = initCanvas(); // 获取ctx实例 // 初始化画布 function initCanvas() { const canvas = document.getElementById("canvas"); const ctx = canvas.getContext("2d"); ctx.fillStyle = "#fff"; ctx.fillRect(0, 0, 260, 310); return ctx; } </script> </html>
解析xls文件
readWorkbookFromRemoteFile().then((res) => { // res 为实际解析代码 [{key:'xxxx',value:'xxxx'},....] }); // 读取xls信息,并处理 function readWorkbookFromRemoteFile() { return new Promise((resolve, reject) => { var xhr = new XMLHttpRequest(); xhr.open("get", "http://127.0.0.1:5500/xls.xls", true); xhr.responseType = "arraybuffer"; xhr.onload = (e) => { if (xhr.status == 200) { var data = new Uint8Array(xhr.response); var workbook = XLSX.read(data, { type: "array" }); // 获取实际表格长度(去除表头) let carryLen = 0; for (const key in workbook.Sheets["Sheet"]) { const ele = workbook.Sheets["Sheet"][key]; if (key.includes("A")) { carryLen++; } } // 解析数据 let xls = []; for (let i = 2; i <= carryLen; i++) { let data = workbook.Sheets["Sheet"]; xls.push({ key: data["A" + i].w, value: data["B" + i].w, }); } resolve(xls); } }; xhr.send(); }); }
看到这里肯定也有细心的好奇宝宝问,为啥循环体中的i
为2呢?
答案:因为表格中的A1,B1为表格的第一行,而第一行是表头,要去除
将链接生成为二维码
new Promise((resolve, reject) => { // 生成二维码 QRCode.toDataURL( 'xxxxxxx', { width: 260, height: 260, margin: 3, }, (error, url) => { if (error) console.error(error); const code = new Image(); code.src = url; code.onload = () => { ctx.drawImage(code, 0, 0); resolve(code); }; } );
写入中间logo
return new Promise((resolve, reject) => { const code = new Image(); code.src = "http://127.0.0.1:5500/logo.jpeg"; code.onload = () => { ctx.drawImage(code, 260 / 2 - 20, 260 / 2 - 20, 40, 40); resolve(); }; });
写入底部文字
// 写入编号 ctx.font = "24px Arial"; ctx.fillStyle = "#000"; ctx.textAlign = "center"; ctx.textBaseline = "middle"; ctx.fillText(xls[index].value, 130, 270);
canvas转化为图片,并下载到本地
// 用于预览 let url = document.getElementById("canvas").toDataURL("image/png"); var a = document.createElement("a"); // 生成一个a元素 var event = new MouseEvent("click"); // 创建一个单击事件 a.download = xls[index].value; // 将a的download属性设置为我们想要下载的图片名称,若name不存在则使用‘下载图片名称'作为默认名称 a.href = url; // 将生成的URL设置为a.href属性 a.dispatchEvent(event); // 触发a的单击事件
第一张图片,完成生成
递归调用
我们修改发起逻辑代码,逻辑尾部增加递归调用就好啦
readWorkbookFromRemoteFile().then((res) => { createImg(res, 0); // 递归生成 }); // ...... // 实际生成逻辑 function createImg(xls, index) { new Promise((resolve, reject) => { // 生成二维码 }) .then((res) => { // 生成中间logo }) .then(() => { // 写入编号 }) .then(() => { // 下载图片 }) .then(() => { setTimeout(() => { if (xls.length > index + 1) { ctx.fillStyle = "#fff"; ctx.fillRect(0, 0, 260, 310); // 初始化画布 createImg(xls, index + 1); } }, 20); // 爱惜机器,加个延时,也可以去掉延时,体会机器的极致速度 }); }
最终效果
至此,终于实现了刷刷刷下载图片炫酷效果,此时可以脑部一段很快的rap,如果华佗再世,崇洋可以医治,外邦来学汉字...............
最终生成的文件
到此这篇关于JS实现读取Excel文件内容并生成二维码的文章就介绍到这了,更多相关JS读取Excel文件内容请搜索自由互联以前的文章或继续浏览下面的相关文章希望大家以后多多支持自由互联!