word表格 转换html 并导出.docx和图片(vue)

一、复制word表格转换成html代码 ,在页面中显示 并且能导出

1、导出使用了htmlDocx插件

//1、使用html-docx-js 插件
npm install html-docx-js --save
npm install html2canvas
//2、在页面中引入
import htmlDocx from 'html-docx-js/dist/html-docx.js';
import html2canvas from "html2canvas";

2、开始写代码 (以下代码包含生成word以及生成图片)

特别注意,关于页面大小(A4,页面边距是无法改变的 我多次尝试 没什么效果 我的方法是定义好表格宽高 人工对生成的word进行改变 )

<template><div><div><el-select v-model="typesettingType"><el-option v-for="item in typesettingTypeOptions" :key="item.value" :label="item.label" :value="item.value"> </el-option></el-select><span>请在下面区域粘贴 Word 表格</span></div><div class="controls" style="overflow: hidden">请在此区域粘贴 Word 表格(typesettingType 为页面方向 1为竖向 <span> </span><divstyle="margin: 0 auto; box-sizing: border-box"class="paste-area":style="`width: ${typesettingType == 1 ? 210 : 297}mm !important;padding:25.4mm 19.1mm`"contenteditable="true"id="content"ref="content"@paste="handlePaste"></div></div><el-button @click="exportToWord">导出为 Word</el-button><el-button @click="downloadImage">导出为 图片</el-button></div>
</template><script>
import htmlDocx from 'html-docx-js/dist/html-docx.js';
import html2canvas from 'html2canvas';
export default {data() {return {updatedHtml: '',typesettingType: 1,typesettingTypeOptions: [{ label: '竖向 A4', value: 1, pageWidth: 210 },{ label: '横向 A4', value: 2, pageWidth: 297 }]};},mounted() {this.typesettingType = 1;},methods: {//下载图片downloadImage() {const container = this.$refs.content;const outputDiv = container.querySelector('table');// 使用 html2canvas 将内容转为图片html2canvas(outputDiv, {scale: 1, // 提高图片的分辨率,默认为1,设置为2 3 可以使图片更清晰最多好像到4logging: false, // 禁用日志输出useCORS: true, // 允许跨域图像allowTaint: true // 允许污染 canvas,解决图片链接不可用问题})// 清空现有内容,显示图片.then((canvas) => {// 创建一个图片对象const imgData = canvas.toDataURL('image/png');// 创建一个链接并下载图片const link = document.createElement('a');link.href = imgData;link.download = 'image.png';link.click();});},async handlePaste(event) {setTimeout(() => {var content = this.$refs.content;console.log('content at line 51:', content);if (content) {const updatedHtml = this.setHtmlWord();this.updatedHtml = updatedHtml.match(/<table[\s\S]*?<\/table>/i)[0];this.$refs.content.innerHTML = this.updatedHtml;console.log('updatedHtml at line 114:', this.updatedHtml);}}, 1000);},//判断内容是否有上下标containsSupOrSub(element) {// 如果当前节点是元素节点if (element.nodeType === 1) {// 如果是 <sup> 或 <sub> 标签,返回 trueif (element.tagName === 'SUP' || element.tagName === 'SUB') {return true;}// 否则,递归检查子节点return Array.from(element.childNodes).some((child) => this.containsSupOrSub(child));}// 如果不是元素节点(如文本节点),返回 falsereturn false;},setHtmlWord() {var setHtmlWord = this.$refs.content.outerHTML;const html = setHtmlWord.replace(/<(table)([^>]*style="([^"]*)")/g, // 匹配有 style 属性的 <table>(match, p1, p2, p3) => {const existingStyle = p3;// 如果现有样式不为空,则在原样式后追加新的样式const updatedStyle = existingStyle? existingStyle +`; width: ${this.typesettingType == 1 ? '488' : '736'}pt !important;`: `width: ${this.typesettingType == 1 ? '488' : '736'}pt !important;`;var str = `<table${p2.replace(`style="${existingStyle}"`, `style="${updatedStyle}"`)}`;return str;}).replace(/style="style=;/g, 'style="').replace(/;;/g, ';').replace(/\n/g, '<br>');const container = document.createElement('div');container.innerHTML = html;const AllTd = container.querySelectorAll('table td'); // 获取 <tr> 中的所有 <td> 元素// 遍历所有 <td> 元素,添加上下边框样式AllTd.forEach((td) => {const currentStyle = td.getAttribute('style');if (currentStyle) {td.setAttribute('style',currentStyle +';color:#000000 !important;border-left:none !important;mso-border-left-alt:none !important;border-right:none !important;mso-border-right-alt:none !important;border-top:none;mso-border-top-alt:none !important;border-bottom:none !important;mso-border-bottom-alt:none !important;');} else {td.setAttribute('style','color:#000000 !important;border-left:none !important;mso-border-left-alt:none !important;border-right:none !important;mso-border-right-alt:none !important;border-top:none;mso-border-top-alt:none !important;border-bottom:none !important;mso-border-bottom-alt:none !important;');}});const firstRowTdElements = container.querySelectorAll('tr:first-child td'); // 获取第一个 <tr> 中的所有 <td> 元素// 遍历所有 <td> 元素,添加上下边框样式firstRowTdElements.forEach((td) => {const currentStyle = td.getAttribute('style');if (currentStyle) {td.setAttribute('style',currentStyle +';color:#000000 !important; border-top:1.0000pt solid #000 !important;mso-border-top-alt:0.5000pt solid #000 !important;border-bottom:1.0000pt solid #000 !important;mso-border-bottom-alt:0.5000pt solid #000 !important;');} else {td.setAttribute('style','color:#000000 !important;border-top:1.0000pt solid #000 !important;mso-border-top-alt:0.5000pt solid #000 !important;border-bottom:1.0000pt solid #000 !important;mso-border-bottom-alt:0.5000pt solid #000 !important;');}});const firstRowTdElementsLast = container.querySelectorAll('tr:last-of-type td');// 遍历所有 <td> 元素,添加上下边框样式firstRowTdElementsLast.forEach((td) => {// 获取当前的 style 属性(如果有)const currentStyle = td.getAttribute('style');// 如果已有 style 属性,则追加边框样式;如果没有 style 属性,则设置新的 styleif (currentStyle) {td.setAttribute('style',currentStyle + ';border-bottom:1.0000pt solid #000 !important;mso-border-bottom-alt:0.5000pt solid #000 !important;');} else {td.setAttribute('style','border-bottom:1.0000pt solid #000 !important;mso-border-bottom-alt:0.5000pt solid #000 !important;');}});// 获取修改后的 HTML 内容const updatedHtml = container.innerHTML;return updatedHtml;},//导出wordexportToWord() {const tableHtml = `<html xmlns:w="urn:schemas-microsoft-com:office:word"><body><div align="center">${this.$refs.content.innerHTML}</div></body></html>`;console.log('tableHtml at line 150:', tableHtml);const converted = htmlDocx.asBlob(tableHtml); // 将 HTML 转换为 Word Blob// 触发文件下载const link = document.createElement('a');link.href = URL.createObjectURL(converted);link.download = 'table.docx';link.click();}}
};
</script><style scoped>
table {border-collapse: collapse;width: 100%;margin: 20px 0;
}th,
td {border: 1px dashed #dcdfe6;padding: 8px;text-align: center;position: relative;
}
table {border-top: 2px solid #000;border-bottom: 1px solid #000;
}
th {border-bottom: 1px solid #000;
}
th input,
td input {width: 100%;border: none;outline: none;text-align: center;
}
th input,
td input ::placeholder {color: #aaa !important;
}.controls {margin: 0 0 20px;
}.drag-handle {cursor: move;
}
::v-deep .paste-area {height: auto; /* A4纸高度 */background: white; /* 纸张背景 */box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1); /* 添加阴影模拟纸张浮起 */border: 1px solid #ddd; /* 模拟纸张边框 *//* padding: 25.4mm 19.1mm; //内边距 */box-sizing: border-box; /* 确保内边距不会影响整体尺寸 */transform-origin: top left;
}
::v-deep .paste-area table {/* border-top: 2px solid #000 !important; *//* border-bottom: 1px solid #000 !important; */margin-left: 0 !important;margin-right: 0 !important;margin: 0 auto !important;
}
::v-deep .paste-area table td {border-top: none !important;border-bottom: none !important;border: 1px dashed #dcdfe6 !important;/* display: flex;align-items: center; */
}
::v-deep .paste-area table td p {display: flex;align-items: center;
}
::v-deep .paste-area table .MsoNormal {max-width: 200px !important; /* 限制容器宽度 */word-wrap: break-word !important;overflow-wrap: break-word !important;
}
.text-container {position: relative;padding: 20px;border: 1px solid #ccc;margin: 20px;font-size: 16px;line-height: 1.5;
}
</style>

在处理td内容时 如果内容出现上标下标要进行特殊处理不然会被覆盖

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/62832.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

我们项目要升级到flutter架构的几点原因

一、探索 Flutter打造卓越移动应用的新时代框架 在移动应用开发的世界里&#xff0c;Flutter已经成为了一个炙手可热的话题。诞生于Google的怀抱&#xff0c;Flutter以其独特的优势和理念&#xff0c;正在引领一场全球范围内的应用开发 ** 。本文将深入探讨Flutter项目的特点、…

DM-VIO(ROS)+t265配置运行记录(ubuntu18.04+ros melodic)

在工作中需要对DM-VIO算法进行测试&#xff0c;于是配置并记录了一下&#xff1a; 首先运行ros接口的dm-vio&#xff0c;一定要先配置源码 https://github.com/lukasvst/dm-vio在这个网址把源码下载下来并解压&#xff0c;并安装一下依赖&#xff1a; sudo apt-get install …

基于Java Springboot成人教育APP且微信小程序

一、作品包含 源码数据库设计文档万字PPT全套环境和工具资源部署教程 二、项目技术 前端技术&#xff1a;Html、Css、Js、Vue、Element-ui 数据库&#xff1a;MySQL 后端技术&#xff1a;Java、Spring Boot、MyBatis 三、运行环境 开发工具&#xff1a;IDEA/eclipse 微信…

基于Java Springboot个人财务APP且微信小程序

一、作品包含 源码数据库设计文档万字PPT全套环境和工具资源部署教程 二、项目技术 前端技术&#xff1a;Html、Css、Js、Vue、Element-ui 数据库&#xff1a;MySQL 后端技术&#xff1a;Java、Spring Boot、MyBatis 三、运行环境 开发工具&#xff1a;IDEA/eclipse 微信…

PotPlayer 最新版本支持使用 Whisper 自动识别语音生成字幕

PotPlayer 最新版本支持使用 Whisper 自动识别语音生成字幕 设置使用下载地址 设置 使用 下载地址 https://www.videohelp.com/software/PotPlayer

【软考网工笔记】网络基础理论——传输层

IPSec协议 Internet协议安全性是一种开放标准的框架结构&#xff0c;通过使用加密的安全服务以确保在Internet协议&#xff08;IP&#xff09;网络上进行保密而安全的通讯。 工作在OSI模型的第三层网络层上&#xff0c;使其在单独使用时适于保护基于TCP或UDP的协议&#xff0…

代码随想录算法训练营第三十四天 | 62.不同路径 | 63. 不同路径 II | 343.整数拆分 | 96.不同的二叉搜索树

Day 34 总结 自己实现中遇到哪些困难 动态规划初始化逻辑&#xff1a;初始化哪些依赖并不存在的格子如何解决当前问题&#xff1f; 通过分解成1个&#xff0c;2个或者多个子问题如何解决子问题&#xff1f;再继续分解&#xff0c;直到要解决一个根部问题&#xff0c;通过根部问…

TinyXML2的一些用法

TinyXML2 原始字符串字面量 TinyXML21. XML文档操作1.1 LoadFile(const char* filename)1.2SaveFile(const char* filename)1.3RootElement()1.4Parse(const char* xml) 2.元素操作2.1 FirstChildElement(const char* name nullptr)2.2 NextSiblingElement(const char* name …

Fastify装饰器:增强你的路由处理功能加入日志

Fastify以其出色的性能和扩展性脱颖而出。装饰器是Fastify提供的一个强大功能&#xff0c;它允许开发者在不修改核心代码的情况下&#xff0c;向请求&#xff08;Request&#xff09;和响应&#xff08;Response&#xff09;对象添加自定义属性和方法。本文将通过一个简单的示例…

redis命令行常用的操作及数据备份

redis命令行常用的操作及数据备份 1.连接命令行2.常用的命令3.数据备份恢复4.桌面管理工具 在日常工作中&#xff0c;有时候会需要去查看redis中某个缓存key是否存在、是否过期等情况&#xff1b;因此&#xff0c;记录整理了一些常用的命令&#xff1b; 1.连接命令行 连接到re…

【大数据学习 | Spark-SQL】关于RDD、DataFrame、Dataset对象

1. 概念&#xff1a; RDD&#xff1a; 弹性分布式数据集&#xff1b; DataFrame&#xff1a; DataFrame是一种以RDD为基础的分布式数据集&#xff0c;类似于传统数据库中的二维表格。带有schema元信息&#xff0c;即DataFrame所表示的二维表数据集的每一列都带有名称和类型…

分布式集群下如何做到唯一序列号

优质博文&#xff1a;IT-BLOG-CN 分布式架构下&#xff0c;生成唯一序列号是设计系统常常会遇到的一个问题。例如&#xff0c;数据库使用分库分表的时候&#xff0c;当分成若干个sharding表后&#xff0c;如何能够快速拿到一个唯一序列号&#xff0c;是经常遇到的问题。实现思…

【算法刷题指南】优先级队列

&#x1f308;个人主页&#xff1a; 南桥几晴秋 &#x1f308;C专栏&#xff1a; 南桥谈C &#x1f308;C语言专栏&#xff1a; C语言学习系列 &#x1f308;Linux学习专栏&#xff1a; 南桥谈Linux &#x1f308;数据结构学习专栏&#xff1a; 数据结构杂谈 &#x1f308;数据…

完美解决Python项目中 ReferenceError: weakly-referenced object no longer exists 错误

问题复现 出现 E ReferenceError: weakly-referenced object no longer exists 报错 self <mysql.connector.cursor_cext.CMySQLCursor object at 0x11de2a7d0> operation \n select id, name,occupation,skills \n from resume \n where id %…

C#中的集合初始化器

C#中的集合初始化器是一种简洁的语法&#xff0c;允许在声明集合的同时初始化其元素。这种语法特别适用于初始化实现了IEnumerable接口并具有Add方法的集合类型&#xff0c;如List<T>、Dictionary<TKey, TValue>等。 集合初始化器的基本用法 集合初始化器的基本语…

【Elasticsearch】03-ES RESTFUL使用

1. Mapping操作 # 新增 PUT /hm {"mappings": {"properties": {"info": {"type": "text","analyzer": "ik_smart"},"age": {"type": "byte"},"email": {&quo…

java基础概念46-数据结构1

一、引入 List集合的三种实现类使用了不同的数据结构&#xff01; 二、数据结构的定义 三、常见的数据结构 3-1、栈 特点&#xff1a;先进后出&#xff0c;后进先出。 java内存容器&#xff1a; 3-2、队列 特点&#xff1a;先进先出、后进后出。 栈VS队列-小结 3-3、数组 3-…

第一部分:基础知识 3. 数据类型 --[MySQL轻松入门教程]

第一部分&#xff1a;基础知识 3. 数据类型 --[MySQL轻松入门教程] MySQL 支持多种数据类型&#xff0c;这些数据类型可以分为几大类&#xff1a;数值类型、字符串类型、日期和时间类型、二进制类型以及枚举和集合。每种类型都有其特定的用途和存储需求。以下是 MySQL 中常用的…

Linux的基本操作及虚拟机设置

文章目录 Linux的目录结构Linux中的常见目录 VI和VIM编辑器什么是VI和VIM编辑器VIM的一般模式VIM的编辑模式VIM的命令模式模式间的切换 虚拟机网络配置查看网络信息修改网络配置信息查看和修改主机名服务管理类命令 虚拟机管理操作VMware为虚拟机拍摄快照VMware为虚拟机执行克隆…

【前端】安装hadoop后,前端启动报错,yarn命令

新安装hadoop后&#xff0c;前端启动项目用yarn命令&#xff0c;报错。 报错&#xff1a;系统找不到指定的路径。 No HADOOP_CONF_DIR set. Please specify it either in yarn-env.cmd or in the environment. 解决&#xff1a;删掉hadoop目录下yarn的文件 检查&#xff1a;…