H5 uses exceljs to replace the content according to the template and export the xlsx file

install module

npm install exceljs

npm install weixin-js-sdk

index.vue, download template, write content through exceljs, and return base64, binary stream

<template>
	<view class="content">
		<!-- <image class="logo" src="/static/logo.png"></image>
		<view class="text-area">
			<text class="title" @click="readfile">导出excel</text>
		</view> -->
	</view>
</template>
<script>
	const ExcelJS = require('exceljs')
	import wx from 'weixin-js-sdk';
	export default {
		data() {
			return {
				title: 'Hello'
			}
		},
		onLoad() {
			uni.showLoading({
				title: '文件生成中...'
			})
			this.readfile();
		},
		methods: {

			/**
			 * arrayBuffer转Base64
			 * @param buffer
			 */
			arrayBufferToBase64(buffer) {
				var binary = '';
				var bytes = new Uint8Array(buffer);
				var len = bytes.byteLength;
				for (var i = 0; i < len; i++) {
					binary += String.fromCharCode(bytes[i]);
				}
				return window.btoa(binary);
			},

			/**
			 * 格式化日期
			 * @param {string | number | Date} value 指定日期
			 * @param {string} format 格式化的规则
			 * @example
			 * ```js
			 * formatDate();
			 * formatDate(1603264465956);
			 * formatDate(1603264465956, "h:m:s");
			 * formatDate(1603264465956, "Y年M月D日");
			 * ```
			 */
			dateFormat(value = Date.now(), format = "Y-M-D h:m:s") {
				const formatNumber = n => `0${n}`.slice(-2);
				const date = new Date(value);
				const formatList = ["Y", "M", "D", "h", "m", "s"];
				const resultList = [];
				resultList.push(date.getFullYear().toString());
				resultList.push(formatNumber(date.getMonth() + 1));
				resultList.push(formatNumber(date.getDate()));
				resultList.push(formatNumber(date.getHours()));
				resultList.push(formatNumber(date.getMinutes()));
				resultList.push(formatNumber(date.getSeconds()));
				for (let i = 0; i < resultList.length; i++) {
					format = format.replace(formatList[i], resultList[i]);
				}
				return format;
			},

			//判断是否微信登陆
			isWeiXinLogin() {
				var ua = window.navigator.userAgent.toLowerCase();
				if (ua.match(/MicroMessenger/i) == 'micromessenger') {
					return true; // 微信中打开
				} else {
					return false; // 普通浏览器中打开
				}
			},

			// 下载excel模板,并替换内容,插入文本、图片、超链接、设置文件名称,并导出xlsx格式的excel文件
			readfile() {
				// alert('6:'+wx.toString());

				// 模板文件一定要是xlsx格式,exceljs不支持操作xls
				uni.request({
					url: "../../static/5s.xlsx", // 支持本地、网络路径
					method: 'GET',
					responseType: 'arraybuffer', // 获取二进制文件流
					success: res => {
						let data = res.data;
						console.log(data);


						const workbook = new ExcelJS.Workbook();

						// 前端生成Excel,只能通过load来加载二进制文件流的方法来读取excel中的单元格内容
						// 一个单元格可同时插入多张图片、图片支持超链接

						// load 方法读取 ArrayBuffer 类型 具体参考文档
						workbook.xlsx.load(data).then(() => {
							const sheet = workbook.getWorksheet(1); //获取第一个worksheet
							const imputData = [];

							// // 读取单元格信息,方式一
							// sheet.eachRow((row, idx) => { // Sheet1的内容 按row 读取
							// 	if (row.values && idx > 1) {
							// 		imputData.push({
							// 			row1: row.values[1] ? row.values[1] : '',
							// 			row2: row.values[2] ? row.values[2] : '',
							// 			row3: row.values[3] ? row.values[3] : '',
							// 		});
							// 	}
							// });

							// // 读取单元格信息,方式二
							// sheet.eachRow(function(row, rowNumber) {
							// 	var rowSize = row.cellCount;
							// 	var numValues = row.actualCellCount;
							// 	//console.log("单元格数量/实际数量:"+rowSize+"/"+numValues);
							// 	// cell.type单元格类型:6-公式 ;2-数值;3-字符串
							// 	row.eachCell(function(cell, colNumber) {
							// 		if (cell.type == 6) {
							// 			var value = cell.result;
							// 		} else {
							// 			var value = cell.value;
							// 		}
							// 		console.log('Cell ' + colNumber + ' = ' + cell
							// 			.type + " " + value);
							// 	});
							// });
							// // 这里 imputData 就是 Sheet1中的内容了
							// console.log('读取内容完成:', imputData);

							// 将数据替换到对应的单元格
							// 工具名称
							// sheet.getCell('A1').value = '5S检查表';
							// sheet.getCell('A1').alignment = {
							// 	vertical: 'middle',
							// 	horizontal: 'center'
							// };
							// sheet.getCell('A1').font = {
							// 	size: 16,
							// 	bold: true
							// }
							// 工具名称
							sheet.getCell('A1').value = '5S检查表';
							// 检查人
							sheet.getCell('B2').value = '李志伟';
							// 检查时间
							sheet.getCell('D2').value = '2022-9-8 14:52:56';
							// 检查区域
							sheet.getCell('F2').value = '车间001 > 1号设备';
							// 项数
							sheet.getCell('B3').value = '10';
							// 正常项数
							sheet.getCell('D3').value = '8';
							// 异常项数
							sheet.getCell('F3').value = '2';
							// 正常率
							sheet.getCell('H3').value = '80%';

							// 序号	检查项	点检内容	点检结果	附件
							sheet.getCell('A5').value = '1'; // 序号
							sheet.getCell('B5').value = '取用是否方便?'; // 检查项
							sheet.getCell('C5').value = '很方便'; // 点检内容
							sheet.getCell('D5').value = '√'; // 点检结果
							sheet.getCell('E5').value = ''; // 附件1
							sheet.getCell('F5').value = ''; // 附件2
							sheet.getCell('G5').value = ''; // 附件3
							sheet.getCell('H5').value = ''; // 附件4
							sheet.getCell('I5').value = ''; // 附件5
							sheet.getCell('J5').value = ''; // 附件6
							sheet.getCell('K5').value = ''; // 附件7
							sheet.getCell('L5').value = ''; // 附件8

							sheet.getCell('A6').value = '2'; // 序号
							sheet.getCell('B6').value = '物品是否完好?'; // 检查项
							sheet.getCell('C6').value = '其中一角已损坏'; // 点检内容
							sheet.getCell('D6').value = '×'; // 点检结果



							// sheet.getCell('E6').value = ''; // 附件1
							// 通过 base64  将图像添加到工作簿
							const Attach1Base64Image =
								"";
							const Attach1ImageId = workbook.addImage({
								base64: Attach1Base64Image,
								extension: 'png',
							});
							// 在 E6:E6 上插入图片
							sheet.addImage(Attach1ImageId, {
								tl: {
									col: 5,
									row: 5
								},
								ext: {
									width: 89,
									height: 69
								},
								editAs: 'undefined'
							});
							// 通过 base64  将图像添加到工作簿
							const playBase64Image =
								"";
							const playImageId = workbook.addImage({
								base64: playBase64Image,
								extension: 'png',
							});
							// 在 E6:E6 上插入播放图标及超链接
							sheet.addImage(playImageId, {
								tl: {
									col: 5.4,
									row: 5.9
								},
								ext: {
									width: 30,
									height: 30
								},
								hyperlinks: {
									hyperlink: 'http://www.ttlean.cn',
									tooltip: '点击播放视频'
								},
								editAs: 'oneCell'
							});
							sheet.getCell('F6').value = ''; // 附件2
							sheet.getCell('G6').value = ''; // 附件3
							sheet.getCell('H6').value = ''; // 附件4
							sheet.getCell('I6').value = ''; // 附件5
							sheet.getCell('J6').value = ''; // 附件6
							sheet.getCell('K6').value = ''; // 附件7
							sheet.getCell('L6').value = ''; // 附件8
							console.log('替换内容完成');

							workbook.xlsx.writeBuffer().then(buffer => {
								// console.log('输出二进制文件流完成', buffer);

								// 浏览器中,只能通过base64格式进行下载,所以先要将二进制流转换成base64
								// Excel的base64前缀,必须是xlsx格式的类型
								// 通过创建a标签下载文件,可以设定文件名

								// 浏览器不能识别二进制流,需要转换成Base64
								var base64 = this.arrayBufferToBase64(buffer);
								var download =
									"data:application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;base64," +
									base64;
								var filename = "5s检查表_" + this.dateFormat(Date.now(), 'YMDhms')

								// alert('1:'+buffer);
								// alert('2:'+base64);
								// alert('3:'+download);
								// alert('4:'+filename);

								console.log({
									// buffer: buffer,
									base64: base64,
									// base64_full: download,
								})

								if (this.isWeiXinLogin()) {
									// 微信中打开
									wx.miniProgram.postMessage({
										data: {
											buffer: buffer,
											base64: base64,
											base64_full: download,
										},
									})
									wx.miniProgram.navigateBack();

								} else {
									// 普通浏览器中打开
									const link = document.createElement('a')
									link.style.display = 'none'
									link.href = download
									link.setAttribute(
										'download',
										filename
									)
									document.body.appendChild(link)
									link.click()
								}

							});

						});

					},
				});


			}
		}
	}
</script>

<style>
	.content {
		display: flex;
		flex-direction: column;
		align-items: center;
		justify-content: center;
	}

	.logo {
		height: 200rpx;
		width: 200rpx;
		margin-top: 200rpx;
		margin-left: auto;
		margin-right: auto;
		margin-bottom: 50rpx;
	}

	.text-area {
		display: flex;
		justify-content: center;
	}

	.title {
		font-size: 36rpx;
		color: #8f8f94;
	}
</style>

Guess you like

Origin blog.csdn.net/qq285679784/article/details/126839011