纯前端JavaScript实现Excel IO案例分享


Posted in Javascript onAugust 26, 2016

公司最近要为某国企做一个**统计和管理系统,

具体要求包含
 •Excel导入导出
 •根据导入的数据进行展示报表
 •图表展示(包括柱状图,折线图,饼图),而且还要求要有动画效果,扁平化风格
 •Excel导出,并要提供客户端来管理Excel 文件
 •...
 

要求真多! 

现在总算是完成了,于是将我的经验分析出来。 

--------------------------------------------------------------------------------

在整个项目架构中,首先就要解决Excel导入的问题。 

由于公司没有自己的框架做Excel IO,就只有通过其他渠道了。 

嗯,我在github上找到了一个开源库xlsx,通过npm方式来安装。
 npm install xlsx --save
之后,在自己的html文件里面添加对js文件的引用
 <script src="./node_modules/xlsx/dist/jszip.js"></script>
<script src="./node_modules/xlsx/dist/xlsx.js"></script>
通过FileReader对象将数据以二进制字符串的方式加载到内存中, 

target.addEventListener('drop', function (e) {
 e.preventDefault();
 handleDrop(e.dataTransfer.files[0]);
});
handleDrop = function(){
 var reader = new FileReader();
 reader.onload = function (e) {
 var data = e.target.result;
 ...
 ...
 };
 reader.readAsBinaryString(f);
}

然后我们下来的操作就是要利用库对data数据进行操作了。 

它暴露了一个对象XLSX,通过XLSX的read() 方法就可以将数据读为JSON对象了。

var workbook = XLSX.read(data, { type: 'binary' });
var sheetName = workbook.SheetNames[0];
var sheet = workbook.Sheets[sheetName];

之后,使用键值对的方式再把数据从sheet中取出来放到表格中。

var table = document.createElement('table');
for (var row = 1; ; row++) {
 if (sheet['A' + row] == null) {
  break;
 }
 var tr = document.createElement('tr');

 for (var col = 65; col <= 90; col++) {
  var c = String.fromCharCode(col);// get 'A', 'B', 'C' ... 
  var key = '' + c + row;
  if (sheet[key] == null) {
   break;
  }
  var td = document.createElement('td');
  td.innerHTML = sheet[key]['w'];
  tr.appendChild(td);
 }
 table.appendChild(tr);
}
document.querySelector('#target').appendChild(table);

下面是完整代码:
index.html 

<!DOCTYPE html>
<html lang="en">
<head>
 <meta charset="UTF-8">
 <title>Document</title>
 <style>
  #target {
   height: 400px;
   width: 700px;
   background-color: #f8f8f8;
   margin: 200px auto;
   overflow:hidden;
   border-radius:5px;
   box-shadow:2px 2px 5px #888;
  } 
  .hover::before {
   content: '请将excel文件拖到这里';
   width: 100%;
   height: 100%;
   display: block;
   text-align: center;
   line-height: 400px;
   font-size: 24px;
   font-family: '微软雅黑';
  }
  #target>table{
   height:250px;
   width:400px;
   border:1px solid #ccc;
   border-radius:3px;
   margin:75px auto;
  }
  #target>table td{
   text-align:center;
   border-top:1px solid #ccc;
   border-left:1px solid #ccc;
  }
   #target>table tr:first-child>td{
    border-top:0px solid #ccc;
   }
   #target>table tr>td:first-child{
    border-left:0px solid #ccc;
   }
 </style>
</head>
<body>
 <div id="target" class="hover">
 </div>
 
 <script src="./node_modules/xlsx/dist/jszip.js"></script>
 <script src="./node_modules/xlsx/dist/xlsx.js"></script>
 <script src="index.js"></script>
</body>
</html>

下面是完整js代码 
index.js 

window.addEventListener('load', function () {
 var target = document.querySelector('#target');
 target.addEventListener('dragenter', function () {
  this.classList.remove('hover');
 });
 target.addEventListener('dragleave', function () {
  this.classList.add('hover');
 });
 target.addEventListener('dragover', function (e) {
  this.classList.remove('hover');
  e.preventDefault();
 });

 target.addEventListener('drop', function (e) {
  e.preventDefault();
  handleDrop(e.dataTransfer.files[0]);
 });

});
var handleDrop = function (f) {
 var reader = new FileReader(),
  name = f.name;
 reader.onload = function (e) {
  var data = e.target.result,
   workbook = XLSX.read(data, { type: 'binary' }),
   sheetName = workbook.SheetNames[0],
   sheet = workbook.Sheets[sheetName],
   table = document.createElement('table');
  
  for (var row = 1; ; row++) {
   if (sheet['A' + row] == null) {
    break;
   }
   var tr = document.createElement('tr');

   for (var col = 65; col <= 90; col++) {
    var c = String.fromCharCode(col);// get 'A', 'B', 'C' ... 
    var key = '' + c + row;
    if (sheet[key] == null) {
     break;
    }
    var td = document.createElement('td');
    td.innerHTML = sheet[key]['w'];
    tr.appendChild(td);
   }
   table.appendChild(tr);
  }
  document.querySelector('#target').appendChild(table);
 };
 reader.readAsBinaryString(f);
}

效果如下:

 纯前端JavaScript实现Excel IO案例分享

这样做好像可行,但是我们很快就放弃了。 
弊端太多了。
 •这个库现在目前还处于开发阶段,在issues里面还有很多的Bug被提出。这没有办法保证最终网站的稳定性。
 •这个库没有办法导入合并单元格的数据,只能是很死板的按照'A', 'B', 'C'... 和1, 2, 3 坐标来查询数据,而且它要求内部单元格不能为空。
 •更比较不方便的就是,它没有行和列的计数的属性。
 •由于这是为国企做的,所以无法将关键功能依赖于这个star量不是很多的库,降低风险,也是为了网站的安全性。
 •...
 
--------------------------------------------------------------------------------

经过小组探讨,我们决定使用另外一款前端控件,叫做 Wijmo。 
首先,从网站上下载Wijmo包,这个控件没有提供npm和bower等方式。

纯前端JavaScript实现Excel IO案例分享 

然后将我需要的包导入进来 

<script src="./wijmo/dist/controls/wijmo.min.js"></script>
<script src="./wijmo/dist/controls/wijmo.grid.min.js"></script>
<script src="./wijmo/dist/controls/wijmo.grid.detail.min.js"></script>
<script src="./wijmo/dist/controls/wijmo.grid.xlsx.min.js"></script>
<script src="./wijmo/dist/controls/wijmo.xlsx.min.js"></script>

此外,还有引入一个jszip的包,是使用js来解压压缩包的一个库。(由于MS的open xml技术,xlsx文件都可以解压成为xml文件,app.xml 里包含了主要的数据)。
 <script src="./jszip.min.js"></script>
读取文件的操作和上面都是一样的 

var handleDrop = function (file) {
 var reader,
  workbook;
 
 if (file) {
  reader = new FileReader;
  reader.onload = function (e) {
   workbook = new wijmo.xlsx.Workbook(),
    workbook.load(reader.result);
  };
  reader.readAsDataURL(file);
 }
}

通过 
 workbook = new wijmo.xlsx.Workbook();
 workbook.load(reader.result);
这两行代码将excel文件加载到内存 中的workbook对象。 

打印workbook对象

 纯前端JavaScript实现Excel IO案例分享

打印这个对象发现,workbook里面包含sheets数组,每个sheet包含rows数组,每个row包含cells数组,每个cell里面vaule属性就是单元格的值。
这简直太又好了 

下面实现一个函数 getCollectionView ,以对象数组的方式来获取数据

var getCollectionView = function (workbook) {
 var collectionView = [];
 if (workbook) {
  var sheet = workbook.sheets[0],
   header = []; // 列标题数组
  
  for (var i = 0, length = sheet.rows.length; i < length; i++) {
   var row = sheet.rows[i],
    rowArray = {};
   for (var j = 0, jLength = row.cells.length; j < jLength; j++) {
    var cell = row.cells[j];
    // 如果是第一行数据,那么是作为列标题出现的,就放进标题数组中
    if (i === 0) {
     header.push(cell.value);
    }
    else {
     // 后面的行数组,就作为rowArray对象的属性存储,属性名就是该列的标题。
     rowArray[header[j]] = cell.value;
    }
   }
   if (i !== 0) {
    collectionView.push(rowArray);
   }
  }
 }
 return collectionView;
}

然后需要一个表格将数据呈现出来,这里我直接使用了Wijmo的FlexGrid表格。 

gridDiv = document.createElement('div');
gridDiv.classList.add('grid');
dataGrid = new wijmo.grid.FlexGrid(gridDiv);// 通过传入容器构造一个FlexGrid表单。
var collectionView = new wijmo.collections.CollectionView(getCollectionView(workbook));
dataGrid.itemsSource = collectionView;

好了,经过上面几个步骤,导入Excel到表格已经实现了 

这是完整的js代码: 

index.js 

(function () {
 var dataGrid = null,
  gridDiv = null,
  workbook = null;
 window.addEventListener('load', function () {
  gridDiv = document.createElement('div');
  gridDiv.classList.add('grid');
  dataGrid = new wijmo.grid.FlexGrid(gridDiv);
  var target = document.querySelector('#target');

  target.addEventListener('dragenter', function (e) {
   e.preventDefault();
   this.classList.remove('hover');

  });
  target.addEventListener('dragleave', function (e) {
   e.preventDefault();
   this.classList.add('hover');
  });
  target.addEventListener('dragover', function (e) {
   e.preventDefault();
   this.classList.remove('hover');
  });

  target.addEventListener('drop', function (e) {
   e.preventDefault();
   handleDrop(e.dataTransfer.files[0]);
   // 将这个表单添加到页面上
   this.appendChild(gridDiv);
  });
 });

 var handleDrop = function (file) {
  var reader;
  var workbook;
  
  if (file) {
   reader = new FileReader;
   reader.onload = function (e) {
    workbook = new wijmo.xlsx.Workbook();
    workbook.load(reader.result);
    var collectionView = new   wijmo.collections.CollectionView(getCollectionView(workbook));
    dataGrid.itemsSource = collectionView;
    // console.log(dataGrid.collectionView);
   };
   reader.readAsDataURL(file);
  }
 }

 var getCollectionView = function (workbook) {
  var collectionView = [];
  
  if (workbook) {
   var sheet = workbook.sheets[0];
   var title = [];
   
   for (var i = 0, length = sheet.rows.length; i < length; i++) {
    var row = sheet.rows[i];
    var rowArray = {};
    
    for (var j = 0, jLength = row.cells.length; j < jLength; j++) {
     var cell = row.cells[j];
     if (i === 0) {
      header.push(cell.value);
     }
     else {
      rowArray[header[j]] = cell.value;
     }
    }
    if (i !== 0) {
     collectionView.push(rowArray);
    }
   }
  }
  return collectionView;
 }
})(window);

下面是效果

 纯前端JavaScript实现Excel IO案例分享

Excel 导出 

欧了 

两句代码实现Excel导出功能 

wijmo.grid.xlsx.FlexGridXlsxConverter.save(dataGrid,
    { includeColumnHeaders: true }, fileName);
这个表格还支持过滤,分组,筛选,编辑。 

面积图和柱状图 

就在完成Excel IO 之后,发现这个控件包还可以做面积图,柱状图和其他很多类型的图形。
所以在这里就演示一个面积图的和一个柱状图的例子。
首先,要将包引进来。 
<script src="./wijmo/dist/controls/wijmo.chart.min.js"></script>
然后经过下面几句代码,就可以使用在页面中插入一个柱状图 

chart = new wijmo.chart.FlexChart('#chart');
chart.initialize({
 itemsSource: collectionView,
 bindingX: 'name',
 options: {
  groupWidth: 15
 },
 series: [
  { name: '年龄', binding: 'age' },
 ]
});

下面看效果

 纯前端JavaScript实现Excel IO案例分享

其中,颜色和柱状图的形状可以调整的。当鼠标移到元素上,还有会小提示。 

在这里,只需要改变一下chart的类型,就可以切换为其他类型的图表 

chart.chartType = chart.chartType === wijmo.chart.ChartType.Column ?
 wijmo.chart.ChartType.Area :
 wijmo.chart.ChartType.Column;

关于本篇的代码已经上传 http://xiazai.3water.com/201608/yuanma/js-xlsWijmo-IO(3water.com).rar

后续会托管到github.

最终还是比较快的完成了任务。 

关于这个项目的Excel IO 就简单介绍到这里,这个项目现在已经完成了,后续会分享一些其他的技术细节。 

希望可以对你提供帮助。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
javascript日期转换 时间戳转日期格式
Nov 05 Javascript
在JavaScript并非所有的一切都是对象
Apr 11 Javascript
JavaScript移除数组元素减少长度的方法
Sep 05 Javascript
jQuery实现的类似淘宝网站搜索框样式代码分享
Aug 24 Javascript
js实现3D图片逐张轮播幻灯片特效代码分享
Sep 09 Javascript
jQuery tagsinput在h5邮件客户端中应用详解
Sep 26 Javascript
Vuejs 用$emit与$on来进行兄弟组件之间的数据传输通信
Feb 23 Javascript
推荐三款日期选择插件(My97DatePicker、jquery.datepicker、Mobiscroll)
Apr 21 jQuery
JS实现中文汉字按拼音排序的方法
Oct 09 Javascript
详解如何使用PM2将Node.js的集群变得更加容易
Nov 15 Javascript
JavaScript的词法结构精华篇
Oct 17 Javascript
vue的hash值原理也是table切换实例代码
Dec 14 Vue.js
js时间比较 js计算时间差的简单实现方法
Aug 26 #Javascript
JavaScript toUpperCase()方法使用详解
Aug 26 #Javascript
js遍历map javaScript遍历map的简单实现
Aug 26 #Javascript
谈谈JavaScript的New关键字
Aug 26 #Javascript
JavaScript数组去重由慢到快由繁到简(优化篇)
Aug 26 #Javascript
针对JavaScript中this指向的简单理解
Aug 26 #Javascript
轻松掌握JavaScript代理模式
Aug 26 #Javascript
You might like
第六节 访问属性和方法 [6]
2006/10/09 PHP
linux php mysql数据库备份实现代码
2009/03/10 PHP
微信营销平台系统?刮刮乐的开发
2014/06/10 PHP
功能强大的PHP图片处理类(水印、透明度、旋转)
2015/10/21 PHP
PHP设计模式之建造者模式定义与用法简单示例
2018/08/13 PHP
jquery创建一个ajax关键词数据搜索实现思路
2013/02/26 Javascript
如何判断微信内置浏览器(通过User Agent实现)
2014/09/01 Javascript
初识Node.js
2014/09/03 Javascript
Javascript中的高阶函数介绍
2015/03/15 Javascript
JavaScript实现鼠标滑过处生成气泡的方法
2015/05/16 Javascript
深入理解JavaScript的React框架的原理
2015/07/02 Javascript
jQuery获取页面及个元素高度、宽度的总结——超实用
2015/07/28 Javascript
JS+CSS实现大气的黑色首页导航菜单效果代码
2015/09/10 Javascript
浅谈JavaScript中的对象及Promise对象的实现
2015/11/15 Javascript
使用JavaScript实现弹出层效果的简单实例
2016/05/31 Javascript
Bootstrap前端开发案例二
2016/06/17 Javascript
bootstrap模态框消失问题的解决方法
2016/12/02 Javascript
jQuery插件Echarts实现的渐变色柱状图
2017/03/23 jQuery
深入研究React中setState源码
2017/11/17 Javascript
基于angular6.0实现的一个组件懒加载功能示例
2018/04/12 Javascript
koa router 多文件引入的方法示例
2019/05/22 Javascript
js+springMVC 提交数组数据到后台的实例
2019/09/21 Javascript
Vue中axios拦截器如何单独配置token
2019/12/27 Javascript
[01:14:31]Secret vs VG 2018国际邀请赛淘汰赛BO3 第一场 8.23
2018/08/24 DOTA
[45:44]完美世界DOTA2联赛PWL S2 FTD vs PXG 第一场 11.27
2020/12/01 DOTA
[01:16:28]DOTA2-DPC中国联赛 正赛 iG vs Magma BO3 第二场 2月23日
2021/03/11 DOTA
同时安装Python2 &amp; Python3 cmd下版本自由选择的方法
2017/12/09 Python
python使用pygame框架实现推箱子游戏
2018/11/20 Python
在python环境下运用kafka对数据进行实时传输的方法
2018/12/27 Python
PyCharm 2019.3发布增加了新功能一览
2019/12/08 Python
Python3 集合set入门基础
2020/02/10 Python
浅析Django 接收所有文件,前端展示文件(包括视频,文件,图片)ajax请求
2020/03/09 Python
python 装饰器的使用示例
2020/10/10 Python
2015年幼儿园卫生保健工作总结
2015/05/12 职场文书
新入职员工工作总结
2015/10/15 职场文书
Python合并多张图片成PDF
2021/06/09 Python