Excel.js 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778
  1. // +----------------------------------------------------------------------
  2. // | CRMEB [ CRMEB赋能开发者,助力企业发展 ]
  3. // +----------------------------------------------------------------------
  4. // | Copyright (c) 2016~2022 https://www.crmeb.com All rights reserved.
  5. // +----------------------------------------------------------------------
  6. // | Licensed CRMEB并不是自由软件,未经许可不能去掉CRMEB相关版权
  7. // +----------------------------------------------------------------------
  8. // | Author: CRMEB Team <admin@crmeb.com>
  9. // +----------------------------------------------------------------------
  10. import { saveAs } from 'file-saver';
  11. import XLSX from 'xlsx';
  12. function datenum(v, date1904) {
  13. if (date1904) v += 1462;
  14. let epoch = Date.parse(v);
  15. return (epoch - new Date(Date.UTC(1899, 11, 30))) / (24 * 60 * 60 * 1000);
  16. }
  17. function data2ws(data) {
  18. const ws = {};
  19. const range = { s: { c: 10000000, r: 10000000 }, e: { c: 0, r: 0 } };
  20. for (let R = 0; R !== data.length; ++R) {
  21. for (let C = 0; C !== data[R].length; ++C) {
  22. if (range.s.r > R) range.s.r = R;
  23. if (range.s.c > C) range.s.c = C;
  24. if (range.e.r < R) range.e.r = R;
  25. if (range.e.c < C) range.e.c = C;
  26. const cell = { v: data[R][C] };
  27. if (cell.v == null) continue;
  28. const cellRef = XLSX.utils.encode_cell({ c: C, r: R });
  29. if (typeof cell.v === 'number') cell.t = 'n';
  30. else if (typeof cell.v === 'boolean') cell.t = 'b';
  31. else if (cell.v instanceof Date) {
  32. cell.t = 'n';
  33. cell.z = XLSX.SSF._table[14];
  34. cell.v = datenum(cell.v);
  35. } else cell.t = 's';
  36. ws[cellRef] = cell;
  37. }
  38. }
  39. if (range.s.c < 10000000) ws['!ref'] = XLSX.utils.encode_range(range);
  40. return ws;
  41. }
  42. function Workbook() {
  43. if (!(this instanceof Workbook)) return new Workbook();
  44. this.SheetNames = [];
  45. this.Sheets = {};
  46. }
  47. function s2ab(s) {
  48. const buf = new ArrayBuffer(s.length);
  49. const view = new Uint8Array(buf);
  50. for (let i = 0; i !== s.length; ++i) view[i] = s.charCodeAt(i) & 0xff;
  51. return buf;
  52. }
  53. /*
  54. * th => 表头
  55. * data => 数据
  56. * fileName => 文件名
  57. * fileType => 文件类型
  58. * sheetName => sheet页名
  59. */
  60. export default function toExcel({ th, data, fileName, fileType, sheetName }) {
  61. data.unshift(th);
  62. let wb = new Workbook();
  63. let ws = data2ws(data);
  64. sheetName = sheetName || 'sheet1';
  65. wb.SheetNames.push(sheetName);
  66. wb.Sheets[sheetName] = ws;
  67. fileType = fileType || 'xlsx';
  68. let wbout = XLSX.write(wb, { bookType: fileType, bookSST: false, type: 'binary' });
  69. fileName = fileName || '列表';
  70. saveAs(new Blob([s2ab(wbout)], { type: 'application/octet-stream' }), `${fileName}.${fileType}`);
  71. }