excel.xlsx.js 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. define(function () {
  2. /*! 下载 Excel 文件 */
  3. function excel(data, name) {
  4. if (name.substr(-5).toLowerCase() !== '.xlsx') {
  5. name += '.xlsx';
  6. }
  7. layui.excel.exportExcel(data, name, 'xlsx')
  8. }
  9. /*! 绑定导出的事件 */
  10. excel.bind = function (done, filename) {
  11. $('body').off('click', '[data-form-export]').on('click', '[data-form-export]', function () {
  12. var form = $(this).parents('form');
  13. var name = this.dataset.filename || filename;
  14. var method = this.dataset.method || form.attr('method') || 'get';
  15. var location = this.dataset.excel || this.dataset.formExport || form.attr('action') || '';
  16. excel.load(location, form.serialize(), method).then(function (ret) {
  17. excel(done(ret), name);
  18. }).fail(function (ret) {
  19. $.msg.tips(ret || '文件导出失败');
  20. });
  21. });
  22. };
  23. /*! 加载导出的文档 */
  24. excel.load = function (url, data, method) {
  25. return (function (defer, lists, loaded) {
  26. loaded = $.msg.loading("正在加载 <span data-upload-count>0.00</span>%");
  27. return (lists = []), LoadNextPage(1, 1), defer;
  28. function LoadNextPage(curPage, maxPage, urlParams) {
  29. $('[data-upload-count]').html((curPage / maxPage * 100).toFixed(2));
  30. if (curPage > maxPage) return $.msg.close(loaded), defer.resolve(lists);
  31. urlParams = (url.indexOf('?') > -1 ? '&' : '?') + 'output=json&not_cache_limit=1&limit=100&page=' + curPage;
  32. $.form.load(url + urlParams, data, method, function (ret) {
  33. if (ret.code) {
  34. lists = lists.concat(ret.data.list);
  35. if (ret.data.page) {
  36. LoadNextPage((ret.data.page.current || 1) + 1, ret.data.page.pages || 1);
  37. }
  38. } else {
  39. defer.reject('数据加载异常');
  40. }
  41. return false;
  42. }, false);
  43. }
  44. })($.Deferred());
  45. };
  46. /*! 读取本地的表格文件 */
  47. excel.read = function (file, filterCallback) {
  48. return (function (defer, reader, loaded, Work) {
  49. reader.onload = function (event) {
  50. Work = XLSX.read(event.target.result, {type: 'binary'});
  51. for (var sheet in Work.Sheets) if (Work.Sheets.hasOwnProperty(sheet)) {
  52. var object = {}, data = Work.Sheets[sheet], k = '', as = '';
  53. for (k in data) if ((as = k.match(/^([A-Z]+)(\d+)$/i))) {
  54. object[as[2]] = object[as[2]] || {};
  55. object[as[2]][as[1]] = excel.read.CellToValue(data[k].v);
  56. }
  57. $.msg.close(loaded);
  58. return defer.resolve(filterCallback ? excel.read.filter(object, filterCallback) : object);
  59. }
  60. $.msg.close(loaded)
  61. };
  62. reader.onerror = function () {
  63. defer.reject('读取文件失败');
  64. };
  65. reader.onprogress = function (event) {
  66. defer.notify((event.loaded / event.total).toFixed(4) * 100);
  67. };
  68. if (typeof file === 'object') {
  69. return reader.readAsBinaryString(file), defer.promise();
  70. } else {
  71. defer.reject('只能读取 file 文件对象');
  72. return defer.promise();
  73. }
  74. })($.Deferred(), new FileReader());
  75. };
  76. /*! 直接推送表格内容 */
  77. excel.read.push = function (url, filterCf, filterFn) {
  78. return (function (defer, $input, loaded) {
  79. $input.appendTo($('body')).click();
  80. $input.on('change', function (event) {
  81. if (!event.target.files || event.target.files.length < 1) return $.msg.tips('没有可操作文件');
  82. loaded = $.msg.loading('<span data-load-name>读取</span> <span data-load-count>0.00%</span>');
  83. excel.read(event.target.files[0], filterCf).then(function (items, total, ers, oks, idx) {
  84. if ((total = items.length) < 1) return cleanAll(), $.msg.tips('未读取到有效数据');
  85. return (ers = 0, oks = 0, idx = 0), $('[data-load-name]').html('更新'), doPostItem(idx, items[idx]);
  86. /*! 执行导入的数据 */
  87. function doPostItem(idx, item, data) {
  88. if (idx >= total) {
  89. return cleanAll(), $.msg.success('共处理' + total + '条记录( 成功 ' + oks + ' 条, 失败 ' + ers + ' 条 )', 3, function () {
  90. $.form.reload();
  91. });
  92. } else {
  93. $('[data-load-count]').html((idx * 100 / total).toFixed(2) + '%( 成功 ' + oks + ' 条, 失败 ' + ers + ' 条 )');
  94. /*! 单元数据过滤 */
  95. data = item;
  96. if (filterFn && (data = filterFn(item)) === false) {
  97. return (ers++), doPostItem(idx + 1, items[idx + 1]);
  98. }
  99. /*! 提交单个数据 */
  100. doUpdate(url, data).then(function (ret) {
  101. (ret.code ? oks++ : ers++), doPostItem(idx + 1, items[idx + 1]);
  102. });
  103. }
  104. }
  105. }).progress(function (progress) {
  106. $('[data-load-count]').html(progress + '%')
  107. }).fail(function () {
  108. cleanAll();
  109. });
  110. });
  111. return defer;
  112. /*! 清理文件选择器 */
  113. function cleanAll() {
  114. $input.remove(), $.msg.close(loaded);
  115. }
  116. /*! 队列方式上传数据 */
  117. function doUpdate(url, item) {
  118. return (function (defer) {
  119. return $.form.load(url, item, 'post', function (ret) {
  120. return defer.resolve(ret), false;
  121. }, false), defer.promise();
  122. })($.Deferred());
  123. }
  124. })($.Deferred(), $('<input class="layui-hide" type="file" accept="application/vnd.ms-excel,application/vnd.openxmlformats-officedocument.spreadsheetml.sheet">'));
  125. }
  126. /*! 解析读取的数据 */
  127. excel.read.filter = function (data, cols) {
  128. return (function (items, item, r, c, k) {
  129. for (r in data) if (r <= 1) {
  130. for (c in data[r]) for (k in cols) {
  131. if (data[r][c] === cols[k].name && !cols[k].bind) cols[k].bind = c;
  132. }
  133. } else {
  134. item = {};
  135. for (k in cols) item[k] = excel.read.CellToValue(data[r][cols[k].bind]);
  136. items.push(item);
  137. }
  138. return items
  139. })([]);
  140. }
  141. /*! 表格单元内容转换 */
  142. excel.read.CellToValue = function (v) {
  143. if (typeof v !== 'undefined' && /^\d+\.\d{12}$/.test(v)) {
  144. return LAY_EXCEL.dateCodeFormat(v, 'YYYY-MM-DD HH:ii:ss');
  145. } else {
  146. return typeof v !== 'undefined' ? v : '';
  147. }
  148. }
  149. return excel;
  150. });