vue项目中,纯前端实现jexcel导出(样式+单元格格式)

        免费开源的框架jexcel(jexcel文档),他自带的导出没什么问题,就是格式和单元格的格式不太好,报表中的背景色等都没有,所以决定找个方法做导出报表功能。废话不多说,直接上代码。

        首先,先把封装的导出js文件中引用的方法奉上:

// 去除千分位 num -> 字符串格式

export function delCommafy(num) {

    if ((num + '').trim() === '') {

        return ''

    }

    if (isType(num, 'string')) {

        num = num.replace(/,/gi, '')

    }

    return num

}

// 把reg(255,255,255)转为16进制颜色(无#号)

export function colorHex(color) {

    // RGB颜色值的正则

    var reg = /^(rgb|RGB)/

    if (reg.test(color)) {

        //   var strHex = "#";

        var strHex = ''

        var colorArr = color.replace(/(?:\(|\)|rgb|RGB)*/g, '').split(',')

        for (var i = 0; i < colorArr.length; i++) {

            var hex = Number(colorArr[i]).toString(16)

            if (hex === '0') {

                hex += hex

            }

            strHex += hex

        }

        return strHex

    } else {

        return color

    }

}

        其次就是封装的导出方法了:

import { colorHex, delCommafy } from '@/utils/index'

import XLSX from 'xlsx-style'

/**

 *

 * @param {string} fileNames 导出文件名

 * @param {string} exportType 导出类型(xlsx...)

 * @param {object} jexcel jexcel对象

 * @param {object} jexcelSpread jexcel的spread对象

 */

export function exportReports(fileNames, exportType, jexcel, jexcelSpread) {

    function saveAs(obj, fileName) {

        var tmpa = document.createElement('a')

        tmpa.download = fileName || '下载'

        tmpa.href = URL.createObjectURL(obj)

        tmpa.click()

        setTimeout(function() {

            URL.revokeObjectURL(obj)

        }, 100)

    }

    const wopts = { bookType: 'xlsx', bookSST: true, type: 'binary', cellStyles: true }

    function downloadExl(fileName, type) {

        let datas = jexcelSpread.getData()

        let merges = jexcelSpread.getMerge()

        let styles = jexcelSpread.getStyle()

        let exportData = [] //用来保存转换好的json

        exportData['!merges'] = []

        exportData['!cols'] = []

        exportData['!rows'] = []

        datas.forEach((rowData, rowIndex) => {

            rowData.forEach((colData, colIndex) => {

                exportData[jexcel.getColumnNameFromId([colIndex, rowIndex])] = {

                    v: Number(delCommafy(colData)) ? delCommafy(colData) : colData,

                }

            })

        })

        for (const key in merges) {

            let startMergeId = jexcel.getIdFromColumnName(key).split('-')

            let startCol = Number(startMergeId[0])

            let startRow = Number(startMergeId[1])

            let endCol = startCol + (merges[key][0] - 1)

            let endRow = startRow + (merges[key][1] - 1)

            exportData['!merges'].push({

                s: {

                    r: startRow,

                    c: startCol,

                },

                e: {

                    r: endRow,

                    c: endCol,

                },

            })

        }

        for (const key in styles) {

            let styleArr = styles[key].split(';').map(item => item.replaceAll(' ', ''))

            let fgRgb = styleArr.filter(item => item.indexOf('background-color') === 0).length

                ? colorHex(

                      styleArr

                          .filter(item => item.indexOf('background-color') === 0)[0]

                          .replace('background-color:', ''),

                  )

                : 'FFFFFF'

            const borderAll = {

                //单元格外侧框线

                top: {

                    style: 'thin',

                    color: { rgb: colorHex('rgb(204,204,204)') },

                },

                bottom: {

                    style: 'thin',

                    color: { rgb: colorHex('rgb(204,204,204)') },

                },

                left: {

                    style: 'thin',

                    color: { rgb: colorHex('rgb(204,204,204)') },

                },

                right: {

                    style: 'thin',

                    color: { rgb: colorHex('rgb(204,204,204)') },

                },

            }

            exportData[key].s = {

                border: borderAll,

                fill: {

                    bgColor: { indexed: 64 },

                    fgColor: { rgb: fgRgb },

                },

                alignment: {

                    vertical: 'center',

                    wrapText: true,

                },

            }

            // 设置单元格类型(n:数值,s:字符串)

            let judgeCondition =

                (String(exportData[key].v).includes('.') && Number(exportData[key].v)) || !exportData[key].v

            exportData[key].t = judgeCondition ? 'n' : 's'

            // 数值格式(二选一)

            exportData[key].z = judgeCondition ? '#,##0.00;-#,##0.00' : ''

        }

        let colsLen = jexcelSpread.colgroup.length

        for (let j = 0; j < colsLen; j++) {

            exportData['!cols'].push({ wpx: Number(jexcelSpread.getWidth()[j]) })

        }

        exportData['!cols'].push({ wpx: 100 })

        let outputPos = Object.keys(exportData) //设置区域,比如表格从A1到D10

        var tmpWB = {

            SheetNames: ['mySheet'], //保存的表标题

            Sheets: {

                mySheet: Object.assign(

                    {},

                    exportData, //内容

                    {

                        '!ref': outputPos[3] + ':' + outputPos[outputPos.length - 1], //设置填充区域

                    },

                ),

            },

        }

        let tmpDown = new Blob(

            [

                s2ab(

                    XLSX.write(

                        tmpWB,

                        { bookType: type == undefined ? 'xlsx' : type, bookSST: false, type: 'binary' }, //这里的数据是用来定义导出的格式类型

                    ),

                ),

            ],

            {

                type: '',

            },

        )

        saveAs(tmpDown, fileName + '.' + (wopts.bookType == 'biff2' ? 'xlsx' : wopts.bookType))

    }

    function s2ab(s) {

        if (typeof ArrayBuffer !== 'undefined') {

            let buf = new ArrayBuffer(s.length)

            let view = new Uint8Array(buf)

            for (let i = 0; i != s.length; ++i) view[i] = s.charCodeAt(i) & 0xff

            return buf

        } else {

            let buf = new Array(s.length)

            for (let i = 0; i != s.length; ++i) buf[i] = s.charCodeAt(i) & 0xff

            return buf

        }

    }

    downloadExl(fileNames, exportType)

}

        vue中引入传参调用就可以了:

                      {

                            type: 'i',

                            content: 'cloud_download',

                            onclick: (element, instance) => {

                                let fileNames = '导出'

                                exportReports(fileNames, 'xlsx', jexcel, instance)

                            },

                        },

        报表原来的页面展示:


        导出效果(背景色及单元格格式都有):


最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,616评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,020评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,078评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,040评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,154评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,265评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,298评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,072评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,491评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,795评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,970评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,654评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,272评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,985评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,815评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,852评论 2 351

推荐阅读更多精彩内容