• 设为首页
  • 收藏本站
  • 积分充值
  • VIP赞助
  • 手机版
  • 微博
  • 微信
    微信公众号 添加方式:
    1:搜索微信号(888888
    2:扫描左侧二维码
  • 快捷导航
    福建二哥 门户 查看主题

    Vue3项目中通过LuckySheet实现Excel在线编辑功能

    发布者: 浪子 | 发布时间: 2025-6-16 07:42| 查看数: 162| 评论数: 0|帖子模式

    一、场景

    在实现Excel文件导入时,领导要求实现在前端导入文件后,不调用后端的接口,而是直接显示excel文件的内容,等待用户修改完以后,再调用后端接口进行文件的提交。在这种应用场景下,使用了LuckySheet进行改功能的实现。
    附上LuckySheet官网地址

    二、Vue3中的基本使用


    1、引入luckySheet

    引入方式有两种,分别是CDN引入与本地引入,下面都会做介绍:

    (1)CDN引入

    在你的index.html中复制下面这段代码,实现CDN引入。
    1. <link rel='stylesheet' href='https://cdn.jsdelivr.net/npm/luckysheet/dist/plugins/css/pluginsCss.css' />
    2. <link rel='stylesheet' href='https://cdn.jsdelivr.net/npm/luckysheet/dist/plugins/plugins.css' />
    3. <link rel='stylesheet' href='https://cdn.jsdelivr.net/npm/luckysheet/dist/css/luckysheet.css' />
    4. <link rel='stylesheet' href='https://cdn.jsdelivr.net/npm/luckysheet/dist/assets/iconfont/iconfont.css' />
    5. <script src="https://cdn.jsdelivr.net/npm/luckysheet/dist/plugins/js/plugin.js"></script>
    6. <script src="https://cdn.jsdelivr.net/npm/luckysheet/dist/luckysheet.umd.js"></script>
    复制代码
    注意,https://cdn.jsdelivr.net/npm/luckysheet/dist/luckysheet.umd.js这个路径意思是会拉取到最新的luckysheet代码,但是如果Luckysheet刚刚发布,jsdelivr网站可能还没来得及从npm上同步过去,故而使用这个路径还是会拉到上一个版本,我们推荐您直接指定最新版本。
    想要指定Luckysheet版本,请在所有的CDN依赖文件后面加上版本号,如:https://cdn.jsdelivr.net/npm/luckysheet@2.1.12/dist/luckysheet.umd.js。
    1. 如何知道最新版本是哪一版?查看最新 <a href="https://github.com/dream-num/Luckysheet/releases" rel="external nofollow" target="_blank">release记录 (opens new window)</a>或者 <a href="https://github.com/dream-num/Luckysheet/blob/master/package.json" rel="external nofollow" target="_blank">package.json (opens new window)</a>的[code]version
    复制代码
    字段。[/code]如果不方便访问 jsdelivr.net,还可以采用本地方式引入。

    (2)本地引入

    本地引入是将CDN引入的文件都放在本地文件中
    第一步:克隆或者下载下面的代码
    1. git clone https://github.com/dream-num/Luckysheet.git
    复制代码
    第二步:安装依赖
    1. npm install
    2. npm install gulp -g  
    复制代码
    第三步:运行查看
    1. npm run dev
    复制代码
    运行效果

    第四步:打包
    1. npm run build
    复制代码
    第五步:本地引入
    打包执行成功后,在文件夹目录下会出现dis文件夹,如下图所示:

    把dist文件夹中的代码全部复制粘贴到你项目的public文件夹中,index.html文件除外。
    在你项目的index.html文件中引入如下代码,如果你复制的位置是其他地方,需要用绝对路径引入这些文件。
    1. <link rel='stylesheet' href='./public/plugins/css/pluginsCss.css' />
    2. <link rel='stylesheet' href='./public/plugins/plugins.css' />
    3. <link rel='stylesheet' href='./public/css/luckysheet.css' />
    4. <link rel='stylesheet' href='./public/assets/iconfont/iconfont.css' />
    5. <script src="./public/plugins/js/plugin.js"></script>
    6. <script src="./public/luckysheet.umd.js"></script>               
    复制代码
    2、在页面中进行展示


    (1)指定一个容器
    1. <div id="luckysheet" class="luckysheet-wrap"></div>

    2. <style scoped>
    3. .luckysheet-wrap {
    4.   margin: 0px;
    5.   padding: 0px;
    6.   position: absolute;
    7.   width: 100%;
    8.   height: 100%;
    9.   left: 0px;
    10.   top: 0px;
    11. }
    12. </style>
    复制代码
    (2)创建一个表格
    1. onMounted(() => {
    2.   //如果这里luckysheet.create报错
    3.   //请使用 window.luckysheet.create
    4.   luckysheet.create({
    5.     container: 'luckysheet'//这里需要和容器的id名称一致
    6.   })
    7. })
    复制代码
    三、Excel文件的导入与导出


    1、导入Excel文件

    安装luckyexcel
    1. npm install luckyexcel --save
    复制代码
    准备一个导入按钮,也可以使用elementUI的导入组件
    这边准备了一个导入按钮
    1. <input id="uploadBtn" type="file" @change="loadExcel" />
    复制代码
    1. const loadExcel = (evt) => {
    2.   const files = evt.target.files
    3.   if (files == null || files.length == 0) {
    4.     alert('请上传文件')
    5.     return
    6.   }

    7.   let name = files[0].name
    8.   let suffixArr = name.split('.'),
    9.     suffix = suffixArr[suffixArr.length - 1]
    10.   if (suffix != 'xlsx') {
    11.     alert('只能导入xlsx文件格式的Excel')
    12.     return
    13.   }
    14.   LuckyExcel.transformExcelToLucky(files[0], function (exportJson, luckysheetfile) {
    15.     if (exportJson.sheets == null || exportJson.sheets.length == 0) {
    16.       alert('导入失败!')
    17.       return
    18.     }
    19.     window.luckysheet.destroy()
    20.     window.luckysheet.create({
    21.       container: 'luckysheet', //容器的Id
    22.       showinfobar: false,
    23.       data: exportJson.sheets,
    24.       title: exportJson.info.name,
    25.       userInfo: exportJson.info.name.creator
    26.     })
    27.   })
    28. }
    复制代码
    2、导出Excel

    导出需要用到excejs和file-saver
    1. //安装exceljs
    2. npm i exceljs --save
    3. //安装文件保存的库
    4. npm i file-saver --save
    复制代码
    这里附上LuckySheet官网提供的导出函数,直接在项目中新建export.js使用
    1. // import { createCellPos } from './translateNumToLetter'
    2. import Excel from 'exceljs'

    3. import FileSaver from 'file-saver'

    4. const exportExcel = function(luckysheet, value) {
    5.   // 参数为luckysheet.getluckysheetfile()获取的对象
    6.   // 1.创建工作簿,可以为工作簿添加属性
    7.   const workbook = new Excel.Workbook()
    8.   // 2.创建表格,第二个参数可以配置创建什么样的工作表
    9.   if (Object.prototype.toString.call(luckysheet) === '[object Object]') {
    10.     luckysheet = [luckysheet]
    11.   }
    12.   luckysheet.forEach(function(table) {
    13.     if (table.data.length === 0) return  true
    14.     // ws.getCell('B2').fill = fills.
    15.     const worksheet = workbook.addWorksheet(table.name)
    16.     const merge = (table.config && table.config.merge) || {}
    17.     const borderInfo = (table.config && table.config.borderInfo) || {}
    18.     // 3.设置单元格合并,设置单元格边框,设置单元格样式,设置值
    19.     setStyleAndValue(table.data, worksheet)
    20.     setMerge(merge, worksheet)
    21.     setBorder(borderInfo, worksheet)
    22.     return true
    23.   })

    24.   // return
    25.   // 4.写入 buffer
    26.   const buffer = workbook.xlsx.writeBuffer().then(data => {
    27.     // console.log('data', data)
    28.     const blob = new Blob([data], {
    29.       type: 'application/vnd.ms-excel;charset=utf-8'
    30.     })
    31.     console.log("导出成功!")
    32.     FileSaver.saveAs(blob, `${value}.xlsx`)
    33.   })
    34.   return buffer
    35. }

    36. var setMerge = function(luckyMerge = {}, worksheet) {
    37.   const mergearr = Object.values(luckyMerge)
    38.   mergearr.forEach(function(elem) {
    39.     // elem格式:{r: 0, c: 0, rs: 1, cs: 2}
    40.     // 按开始行,开始列,结束行,结束列合并(相当于 K10:M12)
    41.     worksheet.mergeCells(
    42.       elem.r + 1,
    43.       elem.c + 1,
    44.       elem.r + elem.rs,
    45.       elem.c + elem.cs
    46.     )
    47.   })
    48. }

    49. var setBorder = function(luckyBorderInfo, worksheet) {
    50.   if (!Array.isArray(luckyBorderInfo)) return
    51.   // console.log('luckyBorderInfo', luckyBorderInfo)
    52.   luckyBorderInfo.forEach(function(elem) {
    53.     // 现在只兼容到borderType 为range的情况
    54.     // console.log('ele', elem)
    55.     if (elem.rangeType === 'range') {
    56.       let border = borderConvert(elem.borderType, elem.style, elem.color)
    57.       let rang = elem.range[0]
    58.       // console.log('range', rang)
    59.       let row = rang.row
    60.       let column = rang.column
    61.       for (let i = row[0] + 1; i < row[1] + 2; i++) {
    62.         for (let y = column[0] + 1; y < column[1] + 2; y++) {
    63.           worksheet.getCell(i, y).border = border
    64.         }
    65.       }
    66.     }
    67.     if (elem.rangeType === 'cell') {
    68.       // col_index: 2
    69.       // row_index: 1
    70.       // b: {
    71.       //   color: '#d0d4e3'
    72.       //   style: 1
    73.       // }
    74.       const { col_index, row_index } = elem.value
    75.       const borderData = Object.assign({}, elem.value)
    76.       delete borderData.col_index
    77.       delete borderData.row_index
    78.       let border = addborderToCell(borderData, row_index, col_index)
    79.       // console.log('bordre', border, borderData)
    80.       worksheet.getCell(row_index + 1, col_index + 1).border = border
    81.     }
    82.     // console.log(rang.column_focus + 1, rang.row_focus + 1)
    83.     // worksheet.getCell(rang.row_focus + 1, rang.column_focus + 1).border = border
    84.   })
    85. }
    86. var setStyleAndValue = function(cellArr, worksheet) {
    87.   if (!Array.isArray(cellArr)) return
    88.   cellArr.forEach(function(row, rowid) {
    89.     row.every(function(cell, columnid) {
    90.       if (!cell) return true
    91.       let fill = fillConvert(cell.bg)

    92.       let font = fontConvert(
    93.         cell.ff,
    94.         cell.fc,
    95.         cell.bl,
    96.         cell.it,
    97.         cell.fs,
    98.         cell.cl,
    99.         cell.ul
    100.       )
    101.       let alignment = alignmentConvert(cell.vt, cell.ht, cell.tb, cell.tr)
    102.       let value = ''

    103.       if (cell.f) {
    104.         value = { formula: cell.f, result: cell.v }
    105.       } else if (!cell.v && cell.ct && cell.ct.s) {
    106.         // xls转为xlsx之后,内部存在不同的格式,都会进到富文本里,即值不存在与cell.v,而是存在于cell.ct.s之后
    107.         // value = cell.ct.s[0].v
    108.         cell.ct.s.forEach(arr => {
    109.           value += arr.v
    110.         })
    111.       } else {
    112.         value = cell.v
    113.       }
    114.       //  style 填入到_value中可以实现填充色
    115.       let letter = createCellPos(columnid)
    116.       let target = worksheet.getCell(letter + (rowid + 1))
    117.       // console.log('1233', letter + (rowid + 1))
    118.       for (const key in fill) {
    119.         target.fill = fill
    120.         break
    121.       }
    122.       target.font = font
    123.       target.alignment = alignment
    124.       target.value = value

    125.       return true
    126.     })
    127.   })
    128. }

    129. var fillConvert = function(bg) {
    130.   if (!bg) {
    131.     return {}
    132.   }
    133.   // const bgc = bg.replace('#', '')
    134.   let fill = {
    135.     type: 'pattern',
    136.     pattern: 'solid',
    137.     fgColor: { argb: bg.replace('#', '') }
    138.   }
    139.   return fill
    140. }

    141. var fontConvert = function(
    142.   ff = 0,
    143.   fc = '#000000',
    144.   bl = 0,
    145.   it = 0,
    146.   fs = 10,
    147.   cl = 0,
    148.   ul = 0
    149. ) {
    150.   // luckysheet:ff(样式), fc(颜色), bl(粗体), it(斜体), fs(大小), cl(删除线), ul(下划线)
    151.   const luckyToExcel = {
    152.     0: '微软雅黑',
    153.     1: '宋体(Song)',
    154.     2: '黑体(ST Heiti)',
    155.     3: '楷体(ST Kaiti)',
    156.     4: '仿宋(ST FangSong)',
    157.     5: '新宋体(ST Song)',
    158.     6: '华文新魏',
    159.     7: '华文行楷',
    160.     8: '华文隶书',
    161.     9: 'Arial',
    162.     10: 'Times New Roman ',
    163.     11: 'Tahoma ',
    164.     12: 'Verdana',
    165.     num2bl: function(num) {
    166.       return num === 0 ? false : true
    167.     }
    168.   }
    169.   // 出现Bug,导入的时候ff为luckyToExcel的val

    170.   let font = {
    171.     name: typeof ff === 'number' ? luckyToExcel[ff] : ff,
    172.     family: 1,
    173.     size: fs,
    174.     color: { argb: fc.replace('#', '') },
    175.     bold: luckyToExcel.num2bl(bl),
    176.     italic: luckyToExcel.num2bl(it),
    177.     underline: luckyToExcel.num2bl(ul),
    178.     strike: luckyToExcel.num2bl(cl)
    179.   }

    180.   return font
    181. }

    182. var alignmentConvert = function(
    183.   vt = 'default',
    184.   ht = 'default',
    185.   tb = 'default',
    186.   tr = 'default'
    187. ) {
    188.   // luckysheet:vt(垂直), ht(水平), tb(换行), tr(旋转)
    189.   const luckyToExcel = {
    190.     vertical: {
    191.       0: 'middle',
    192.       1: 'top',
    193.       2: 'bottom',
    194.       default: 'top'
    195.     },
    196.     horizontal: {
    197.       0: 'center',
    198.       1: 'left',
    199.       2: 'right',
    200.       default: 'left'
    201.     },
    202.     wrapText: {
    203.       0: false,
    204.       1: false,
    205.       2: true,
    206.       default: false
    207.     },
    208.     textRotation: {
    209.       0: 0,
    210.       1: 45,
    211.       2: -45,
    212.       3: 'vertical',
    213.       4: 90,
    214.       5: -90,
    215.       default: 0
    216.     }
    217.   }

    218.   let alignment = {
    219.     vertical: luckyToExcel.vertical[vt],
    220.     horizontal: luckyToExcel.horizontal[ht],
    221.     wrapText: luckyToExcel.wrapText[tb],
    222.     textRotation: luckyToExcel.textRotation[tr]
    223.   }
    224.   return alignment
    225. }

    226. var borderConvert = function(borderType, style = 1, color = '#000') {
    227.   // 对应luckysheet的config中borderinfo的的参数
    228.   if (!borderType) {
    229.     return {}
    230.   }
    231.   const luckyToExcel = {
    232.     type: {
    233.       'border-all': 'all',
    234.       'border-top': 'top',
    235.       'border-right': 'right',
    236.       'border-bottom': 'bottom',
    237.       'border-left': 'left'
    238.     },
    239.     style: {
    240.       0: 'none',
    241.       1: 'thin',
    242.       2: 'hair',
    243.       3: 'dotted',
    244.       4: 'dashDot', // 'Dashed',
    245.       5: 'dashDot',
    246.       6: 'dashDotDot',
    247.       7: 'double',
    248.       8: 'medium',
    249.       9: 'mediumDashed',
    250.       10: 'mediumDashDot',
    251.       11: 'mediumDashDotDot',
    252.       12: 'slantDashDot',
    253.       13: 'thick'
    254.     }
    255.   }
    256.   let template = {
    257.     style: luckyToExcel.style[style],
    258.     color: { argb: color.replace('#', '') }
    259.   }
    260.   let border = {}
    261.   if (luckyToExcel.type[borderType] === 'all') {
    262.     border['top'] = template
    263.     border['right'] = template
    264.     border['bottom'] = template
    265.     border['left'] = template
    266.   } else {
    267.     border[luckyToExcel.type[borderType]] = template
    268.   }
    269.   // console.log('border', border)
    270.   return border
    271. }

    272. function addborderToCell(borders, row_index, col_index) {
    273.   let border = {}
    274.   const luckyExcel = {
    275.     type: {
    276.       l: 'left',
    277.       r: 'right',
    278.       b: 'bottom',
    279.       t: 'top'
    280.     },
    281.     style: {
    282.       0: 'none',
    283.       1: 'thin',
    284.       2: 'hair',
    285.       3: 'dotted',
    286.       4: 'dashDot', // 'Dashed',
    287.       5: 'dashDot',
    288.       6: 'dashDotDot',
    289.       7: 'double',
    290.       8: 'medium',
    291.       9: 'mediumDashed',
    292.       10: 'mediumDashDot',
    293.       11: 'mediumDashDotDot',
    294.       12: 'slantDashDot',
    295.       13: 'thick'
    296.     }
    297.   }
    298.   // console.log('borders', borders)
    299.   for (const bor in borders) {
    300.     // console.log(bor)
    301.     if (borders[bor].color.indexOf('rgb') === -1) {
    302.       border[luckyExcel.type[bor]] = {
    303.         style: luckyExcel.style[borders[bor].style],
    304.         color: { argb: borders[bor].color.replace('#', '') }
    305.       }
    306.     } else {
    307.       border[luckyExcel.type[bor]] = {
    308.         style: luckyExcel.style[borders[bor].style],
    309.         color: { argb: borders[bor].color }
    310.       }
    311.     }
    312.   }

    313.   return border
    314. }

    315. function createCellPos(n) {
    316.   let ordA = 'A'.charCodeAt(0)

    317.   let ordZ = 'Z'.charCodeAt(0)
    318.   let len = ordZ - ordA + 1
    319.   let s = ''
    320.   while (n >= 0) {
    321.     s = String.fromCharCode((n % len) + ordA) + s

    322.     n = Math.floor(n / len) - 1
    323.   }
    324.   return s
    325. }

    326. export {
    327.   exportExcel
    328. }
    复制代码
    在页面中引入export.js文件
    1. import { exportExcel } from './export'
    复制代码
    在页面中使用下面这句代码进行导出
    1. exportExcel(luckysheet.getAllSheets(), '导出的文件名字')
    复制代码
    四、注意事项

    在导出文件时,如果需要将导出的数据通过接口传输给后端,需要进行文件格式的转换,以file为例。
    1. //获取buffer的
    2. workbook.xlsx.writeBuffer()
    3. .then((arrayBuffer) => {
    4.                   const blob = new Blob(arrayBuffer, {
    5.                                   //导出为xlsx文件格式
    6.                                 type: 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'
    7.                });
    8.               //转为file格式
    9.                   let file = new File([blob], '导出.xlsx', {type: blob.type});
    10.                   /**
    11.                    这里书写接口上传代码
    12.                   */
    13.         }).catch(err=>{
    14.                   dialogVisible.value = false
    15.         })
    复制代码
    LuckySheet现在升级为Univer Docs,如果需要使用Univer Docs进行展示excel,可以参考另一篇文章进行了解。
    以上就是Vue3项目中通过LuckySheet实现Excel在线编辑的详细内容,更多关于Vue3 LuckySheet在线编辑Excel的资料请关注脚本之家其它相关文章!

    来源:https://www.jb51.net/javascript/339448a6b.htm
    免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

    本帖子中包含更多资源

    您需要 登录 才可以下载或查看,没有账号?立即注册

    ×

    最新评论

    QQ Archiver 手机版 小黑屋 福建二哥 ( 闽ICP备2022004717号|闽公网安备35052402000345号 )

    Powered by Discuz! X3.5 © 2001-2023

    快速回复 返回顶部 返回列表