解析99行代码的在线电子表格(Web Spreadsheet in 99 lines)

序言

随着浏览器运行性能及前端技术的日新月异,对于使用在线表格做报表已经成为时下主流趋势,而在线电子表格也层出不穷,如Google的SpreadSheet等,由于公司报表类产品中需要使用在线电子表格,并且要在基本的电子表格控件上增加许多额外与业务相关的扩展,因此在咨询及使用过一些通用工具后决定自己造轮子。造轮子之前先学习前人的经验,如何设计在线电子表格。为了入门我们先选择了本文将介绍的这个99行代码完成的在线电子表格。

99行代码的电子表格简介

绘制表格的方法

1 定义行数组(Rows)与列数组(Cols

  $scope.Cols = [], $scope.Rows = [];

2 初始化数组

  • 将列数组(Cols)初始化为['A','B','C','D','E','F','G','H']
  • 将行数组(Rows)初始化为[1,2,3,4,...,18,19,20]
  makeRange($scope.Cols, 'A', 'H');
  makeRange($scope.Rows, 1, 20);
  • 其中makeRange(array, cur, end)函数的作用就是根据curend之间的范围初始化给array数组,源码如下:
function makeRange(array, cur, end) {
  while (cur <= end) {
    array.push(cur);
    // If it’s a number, increase it by one; otherwise move to next letter
    cur = (isNaN( cur ) ? String.fromCharCode(cur.charCodeAt()+1 ) : cur+1);
  }
}

3 绑定页面元素

  • 利用AngularJS的双向数据绑定特性,将Javascript变量与页面元素进行绑定
<table>
  <tr>
    <th>
      <button type="button" ng-click="reset();calc()">↻</button>
    </th>
    <th ng-repeat="col in Cols">{{ col }}</th>
  </tr>
  <tr ng-repeat="row in Rows">
    <th>{{ row }}</th>
    <td ng-repeat="col in Cols" ng-class="{ formula: ( '=' === sheet[col+row][0] ) }">
      <input id="{{ col+row }}" ng-model="sheet[col+row]" ng-change="calc()" ng-model-options="{ debounce: 200 }" ng-keydown="keydown( $event, col, row )">
    </td>
  </tr>
</table>

通过该HTML代码可以了解,其实表格是通过<table>标签实现的,其中的单元格就是<input>文本框,通过行列的循环(ng-repeat)绘制出一张电子表格。

  • 列头为表格第一行,列头第一个元素是一个刷新按钮,其余通过循环(ng-repeat)列数组(Cols)绘制,并将数组元素值作为元素显示内容
<tr>
  <th>
    <button type="button" ng-click="reset();calc()">↻</button>
  </th>
  <th ng-repeat="col in Cols">{{ col }}</th>
</tr>
  • 第二行开始绘制表格内单元格,首先按行数组(Rows)进行循环(ng-repeat),然后在单个行循环体内,第一个列作为行头,显示行数组的值作为行号,其余通过循环(ng-repeat)列数组(Cols)并插入<input>文本框,并将列号加行号的组合字符串赋值给该单元格ID属性
<tr ng-repeat="row in Rows">
  <th>{{ row }}</th>
  <td ng-repeat="col in Cols" ng-class="{ formula: ( '=' === sheet[col+row][0] ) }">
    <input id="{{ col+row }}" ng-model="sheet[col+row]" ng-change="calc()" ng-model-options="{ debounce: 200 }" ng-keydown="keydown( $event, col, row )">
  </td>
</tr>

如此,一张电子表格就绘制完成

单元格添加键盘事件

  • HTML中<input>标签上声明键盘按下事件(ng-keydown
    <input id="{{ col+row }}" ng-model="sheet[col+row]" ng-change="calc()" ng-model-options="{ debounce: 200 }" ng-keydown="keydown( $event, col, row )">
  • JS中实现keydown( $event, col, row )方法
// UP(38) and DOWN(40)/ENTER(13) move focus to the row above (-1) and below (+1).
$scope.keydown = function(event, col, row) {  
  switch(event.which) {
    case 38: case 40: case 13: $timeout( function() {
      var direction = (event.which === 38) ? -1 : +1;
      var cell = document.querySelector( '#' + col + (row + direction) );
      if (cell) {
        cell.focus();
      }
    } );
  }
};

如果当键盘按下“上”键(键值:38),则根据单元格ID属性找到上方第一个一个单元格,并使其成为焦点(focus);如果当键盘按下“下”键(键值:40)或“回车”键(键值:13),则根据单元格ID属性找到下方一个单元格,并使其成为焦点(focus)。

表格的数值存储

  • 在Javascript代码中声明sheet对象
// Restore the previous .sheet
$scope.sheet = angular.fromJson( localStorage.getItem( '' ) );
  • 在HTML中<input>文本框标签内绑定(ng-modelsheet变量
  <input id="{{ col+row }}" ng-model="sheet[col+row]" ng-change="calc()" ng-model-options="{ debounce: 200 }" ng-keydown="keydown( $event, col, row )">

sheet对象中存储形式为键值对,如

$scope.sheet = { B1: 1874, A2: '+', B2: 2046, A3: '⇒', B3: '=B1+B2' };

其中键为单元格ID属性,即行+列组合的字符串,值为单元格值。

表格的运算过程

  • 表格计算通过Web Workers开启计算线程完成
$scope.worker = new Worker("/echo/js/?js="+encodeURIComponent("(" + WorkerJS.toString() + ")()"));
  • 具体计算方法写在WorkerJS()函数内
// Worker.js
function WorkerJS () {
  var sheet, errs, vals;
  self.onmessage = function(message) {
    sheet = message.data, errs = {}, vals = {};

    Object.getOwnPropertyNames(sheet || {}).forEach(function(coord) {
      // Four variable names pointing to the same coordinate: A1, a1, $A1, $a1
      [ '', '$' ].forEach(function(p) { [ coord, coord.toLowerCase() ].forEach(function(c){
        var name = p+c;

        // Worker is reused across computations, so only define each variable once
        if ((Object.getOwnPropertyDescriptor( self, name ) || {}).get) { return; }

        // Define self['A1'], which is the same thing as the global variable A1
        Object.defineProperty( self, name, { get: function() {
          if (coord in vals) { return vals[coord]; }
          vals[coord] = NaN;

          // Turn numeric strings into numbers, so =A1+C1 works when both are numbers
          var x = +sheet[coord];
          if (sheet[coord] !== x.toString()) { x = sheet[coord]; }

          // Evaluate formula cells that begin with =
          try { vals[coord] = (('=' === x[0]) ? eval.call( null, x.slice( 1 ) ) : x);
          } catch (e) {
            var match = /\$?[A-Za-z]+[1-9][0-9]*\b/.exec( e );
            if (match && !( match[0] in self )) {
              // The formula refers to a uninitialized cell; set it to 0 and retry
              self[match[0]] = 0;
              delete vals[coord];
              return self[coord];
            }
            // Otherwise, stringify the caught exception in the errs object
            errs[coord] = e.toString();
          }
          // Turn vals[coord] into a string if it's not a number or boolean
          switch (typeof vals[coord]) { case 'function': case 'object': vals[coord]+=''; }
          return vals[coord];
        } } );
      }); });
    });

    // For each coordinate in the sheet, call the property getter defined above
    for (var coord in sheet) { self[coord]; }
    postMessage([ errs, vals ]);
  };
}
  • 计算流程为


总结

经过分析,99行实现的该电子表格虽然功能简单,但是基本的表格绘制及运算实现了,为我们未来设计电子表格结构提供了重要的参考价值。

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

推荐阅读更多精彩内容