如何使用JavaScript实现纯前端读取和导出excel、csv文件

工具类可以直接复制稍微修改进行使用,使用方法参考使用示例,基础的文件导入可以参考参考链接。

  • 使用:
import TableImport from './TableImport'
import TableModelExport from './TableModelExport'
import TableModelExportCSV from './TableModelExportCSV'



  handleImportFromExcelData = (data = []) => {

  }

  render() {

const propsTableImport = {
handleImportFromExcelData:this.handleImportFromExcelData
}

const columns = [
{label:'姓名'},
{label:'单位'},
]

const propsTableModelExport = {
modelName:'模板名称',
columns:columns
}

    return (
      <Fragment>
<TableImport {...propsTableImport} />
<TableModelExport {...propsTableModelExport} />
<TableModelExportCSV {...propsTableModelExport} />
        
      </Fragment>
    )
  }

  • 工具类
    导入Excel、CSV文件内容:
    TableImport.js
import React, { Component } from 'react'
import { Button, Input, message } from 'components'
import uuid from '~/utils/uuid'

export default class TableImport extends Component {
  state = {
    fileId: uuid()
  }

  handleImport = () => {
    document.getElementById(this.state.fileId).click()
  }

  onChangeInputFile = (e) => {
    var files = e.target.files
    if (files.length == 0) return
    var f = files[0]
    if (!(/\.xlsx$/g.test(f.name) || /\.csv$/g.test(f.name))) {
      message.error('仅支持读取xlsx或csv格式!')
      return
    }
    this.readWorkbookFromLocalFile(f, this.readWorkbook)
  }

  // 读取本地excel文件
  readWorkbookFromLocalFile = (file, callback) => {
    const reader = new FileReader()
    reader.onload = function(e) {
      const data = e.target.result
      const workbook = XLSX.read(data, { type: 'binary' })
      if (callback) callback(workbook)
    }
    reader.readAsBinaryString(file)
  }

  readWorkbook = (workbook) => {
    var sheetNames = workbook.SheetNames // 工作表名称集合
    var worksheet = workbook.Sheets[sheetNames[0]] // 这里我们只读取第一张sheet
    var csv = XLSX.utils.sheet_to_csv(worksheet)
    window.csv = csv
    const data = this.csvToArray(csv)
    this.props.handleImportFromExcelData(data)
  }

  csvToArray = (csv) => {
    const rows = csv.split('\n')
    rows.pop() // 最后一行没用的
    const keys = rows[0].split(',')
    const arr = []
    for (let i = 1; i < rows.length; i++) {
      const item = rows[i]
      const values = item.split(',')
      const arrItem = {

      }
      keys.map((key, index) => {
        arrItem[key] = values[index]
      })
      arr.push(arrItem)
    }
    // const data = [
    //   {
    //     源系统名称: '源系统名称',
    //     源IP地址: '10.10.10.10',
    //     源端口: '8080',
    //     目的系统名称: '目的系统名称',
    //     目的IP地址: '10.10.10.10',
    //     目的端口: '8080',
    //     访问协议: 'TCP',
    //     访问操作: '放行'
    //   },
    //   {
    //     源系统名称: '源系统名称',
    //     源IP地址: '10.10.10.10',
    //     源端口: '8080',
    //     目的系统名称: '目的系统名称',
    //     目的IP地址: '10.10.10.10',
    //     目的端口: '8080',
    //     访问协议: 'UDP',
    //     访问操作: '阻断'
    //   }
    // ]

    return arr
  }

  render() {
    return (
      <div style={{ display: 'inline' }}>
        <Button
          style={{ marginTop: 10, marginLeft: 10 }}
          type="primary"
          icon="upload"
          onClick={this.handleImport}
        >
          {i18n('ticket.form.add.import', '导入')}
        </Button>
        <Input type="file" id={this.state.fileId}
          onChange={this.onChangeInputFile}
          style={{ display: 'none' }} accept=".csv,application/vnd.openxmlformats-officedocument.spreadsheetml.sheet" />
      </div>
    )
  }
}

导出Excel文件:
TableModelExport.js

import React, { Component } from 'react'
import { Button, Input, message } from 'components'

export default class TableModelExport extends Component {
  handleExport = () => {
    var csv = '姓名,单位,手机号,可访问资源,VPN账号\n'
    csv = this.props.columns.map((item, i) => {
      return item.label
    })
    csv.join(',')
    csv = csv + '\n'
    this.exportExcel(csv)
  }

  exportExcel = (csv) => {
    const sheet = this.csv2sheet(csv)
    const blob = this.sheet2blob(sheet)
    this.openDownloadDialog(blob, `${this.props.modelName}模板.xlsx`)
  }

  // csv转sheet对象
  csv2sheet=(csv) => {
    const sheet = {} // 将要生成的sheet
    csv = csv.split('\n')
    csv.forEach(function(row, i) {
      row = row.split(',')
      if (i == 0) sheet['!ref'] = 'A1:' + String.fromCharCode(65 + row.length - 1) + (csv.length - 1)
      row.forEach(function(col, j) {
        sheet[String.fromCharCode(65 + j) + (i + 1)] = { v: col }
      })
    })
    return sheet
  }

  // 将一个sheet转成最终的excel文件的blob对象,然后利用URL.createObjectURL下载
  sheet2blob = (sheet, sheetName) => {
    sheetName = sheetName || 'sheet1'
    const workbook = {
      SheetNames: [sheetName],
      Sheets: {}
    }
    workbook.Sheets[sheetName] = sheet
    // 生成excel的配置项
    const wopts = {
      bookType: 'xlsx', // 要生成的文件类型
      bookSST: false, // 是否生成Shared String Table,官方解释是,如果开启生成速度会下降,但在低版本IOS设备上有更好的兼容性
      type: 'binary'
    }
    const wbout = XLSX.write(workbook, wopts)
    const blob = new Blob([s2ab(wbout)], { type: 'application/octet-stream' })
    // 字符串转ArrayBuffer
    function s2ab(s) {
      const buf = new ArrayBuffer(s.length)
      const view = new Uint8Array(buf)
      for (let i = 0; i != s.length; ++i) view[i] = s.charCodeAt(i) & 0xFF
      return buf
    }
    return blob
  }

  /**
   * 通用的打开下载对话框方法,没有测试过具体兼容性
   * @param url 下载地址,也可以是一个blob对象,必选
   * @param saveName 保存文件名,可选
   */
  openDownloadDialog = (url, saveName) => {
    if (typeof url === 'object' && url instanceof Blob) {
      url = URL.createObjectURL(url) // 创建blob地址
    }
    const aLink = document.createElement('a')
    aLink.href = url
    aLink.download = saveName || '' // HTML5新增的属性,指定保存文件名,可以不要后缀,注意,file:///模式下不会生效
    var event
    if (window.MouseEvent) event = new MouseEvent('click')
    else {
      event = document.createEvent('MouseEvents')
      event.initMouseEvent('click', true, false, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null)
    }
    aLink.dispatchEvent(event)
  }

  render() {
    return (
      <div style={{ display: 'inline' }}>
        <Button
          style={{ marginTop: 10, marginLeft: 10 }}
          type="primary"
          icon="download"
          onClick={this.handleExport}
        >
          {i18n('ticket.form.add.exportExcel', '导出Excel模板')}
        </Button>
      </div>
    )
  }
}

导出CSV文件:
TableModelExportCSV.js

import React, { Component } from 'react'
import { Button, Input, message } from 'components'

export default class TableModelExportCSV extends Component {
  handleExport = () => {
    var csv = '姓名,单位,手机号,可访问资源,VPN账号\n'
    csv = this.props.columns.map((item, i) => {
      return item.label
    })
    csv.join(',')
    csv = csv + '\n'
    this.exportCSV(csv)
  }

  exportCSV = (data = '') => {
    // 两个表数组转成字符串合并
    const merged = data
    // console.log(JSON.stringify(merged));//打印结果

    // ## 导出操作
    // encodeURIComponent解决中文乱码
    const uri = 'data:text/csv;charset=utf-8,\ufeff' + encodeURIComponent(merged)
    // 通过创建a标签实现
    const link = document.createElement('a')
    link.href = uri
    // 对下载的文件命名
    link.download = `${this.props.modelName}模板.csv`
    document.body.appendChild(link)
    link.click()
  }

  render() {
    return (
      <div style={{ display: 'inline' }}>
        <Button
          style={{ marginTop: 10, marginLeft: 10 }}
          type="primary"
          icon="download"
          onClick={this.handleExport}
        >
          {i18n('ticket.form.add.exportCSV', '导出CSV模板')}
        </Button>
      </div>
    )
  }
}

参考:
读取Excel、导出Excel、读取CSV:
https://www.cnblogs.com/liuxianan/p/js-excel.html

导出CSV:
https://blog.csdn.net/weixin_41628411/article/details/106280073

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

推荐阅读更多精彩内容