纯前端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 相关文章推荐
HTML代码中标签的全部属性 中文注释说明
Mar 26 Javascript
javascript中简单的进制转换代码实例
Oct 26 Javascript
js 跳出页面的frameset框架示例介绍
Dec 23 Javascript
jquery实现类似EasyUI的页面布局可改变左右的宽度
Sep 12 Javascript
JavaScript实现计算字符串中出现次数最多的字符和出现的次数
Mar 12 Javascript
JS获取Table中td值的方法
Mar 19 Javascript
TypeScript 中接口详解
Jun 19 Javascript
JS中sort函数排序用法实例分析
Jun 16 Javascript
浅谈jQuery this和$(this)的区别及获取$(this)子元素对象的方法
Nov 29 Javascript
如何在Angular2中使用jQuery及其插件的方法
Feb 09 Javascript
AngularJs 禁止模板缓存的方法
Nov 28 Javascript
详解Vue.js 可拖放文本框组件的使用
Mar 03 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
fleaphp常用方法分页之Pager使用方法
2011/04/23 PHP
php计算两个整数的最大公约数常用算法小结
2015/03/05 PHP
PHP实现过滤掉非汉字字符只保留中文字符
2015/06/04 PHP
php实现微信公众号主动推送消息
2015/12/31 PHP
PHP多进程编程之僵尸进程问题的理解
2017/10/15 PHP
使用Laravel中的查询构造器实现增删改查功能
2019/09/03 PHP
laravel 框架执行流程与原理简单分析
2020/02/01 PHP
JavaScript 继承机制的实现(待续)
2010/05/18 Javascript
Jquery 改变radio/checkbox选中状态,获取选中的值(示例代码)
2013/12/12 Javascript
JavaScript实现Iterator模式实例分析
2015/06/09 Javascript
第一次接触神奇的Bootstrap表单
2016/07/27 Javascript
Vue.js 父子组件通讯开发实例
2016/09/06 Javascript
angularJS 指令封装回到顶部示例详解
2017/01/22 Javascript
Vue的土著指令和自定义指令实例详解
2018/02/04 Javascript
深入浅析javascript函数中with
2018/10/28 Javascript
微信小程序云开发之云函数详解
2019/05/16 Javascript
vue element-ui之怎么封装一个自己的组件的详解
2019/05/20 Javascript
vue+django实现一对一聊天功能的实例代码
2019/07/17 Javascript
vuejs移动端实现div拖拽移动
2019/07/25 Javascript
Vue+ElementUI table实现表格分页
2019/12/14 Javascript
js找出5个数中最大的一个数和倒数第二大的数实现方法示例小结
2020/03/04 Javascript
jQuery实现简单飞机大战
2020/07/05 jQuery
[03:40]DOTA2抗疫特别篇《英雄年代》
2020/02/28 DOTA
python if not in 多条件判断代码
2016/09/21 Python
python判断字符串是否是json格式方法分享
2017/11/07 Python
python对list中的每个元素进行某种操作的方法
2018/06/29 Python
在python中利用最小二乘拟合二次抛物线函数的方法
2018/12/29 Python
Python后台开发Django会话控制的实现
2019/04/15 Python
python 基于pygame实现俄罗斯方块
2021/03/02 Python
css3教程之倾斜页面
2014/01/27 HTML / CSS
不拖欠农民工工资承诺书
2014/03/31 职场文书
大学生活自我评价
2014/04/09 职场文书
通信工程专业求职信
2014/06/04 职场文书
机械专业求职信范文
2014/07/15 职场文书
上甘岭观后感
2015/06/10 职场文书
思想工作总结范文
2015/08/12 职场文书