如何解析你,Excel的Date呀

簡單的背景介紹

不久前,咱們接了一個本身作前端後端產品的活,今後過上了可憐巴巴敲代碼開開心心收穫知識的日子呢。javascript

那是一個平平無奇的週一下午

用戶小姐姐在羣裏說,系統篩選工卡有效期很差使。(系統:不不不,不是個人鍋前端

我看了一下數據庫,發現,咱們原定的有效期格式是這樣的java

整整齊齊。數據庫

數據庫中當時的數據是這樣的json

甚至是這樣的後端

看到這種狀況,我以爲確定是輸入的時候輸的不太對(年輕…app

因而我決定從Excel下手this

小姐姐們的操做流程是先用咱們的系統導出一份Excel,編輯以後再導入系統的,那隻要我把這工卡一列的格式限制爲日期,就必定能夠統一格式的,嗯。spa

咱們項目使用了js-xlsx處理表格的導入導出,下面是導出Excel的僞代碼:code

import * as XLSX from 'xlsx';

const xlsxMineType = 'application/vnd.openxmlformats-
officedocument.spreadsheetml.sheet'
const data = 數據.map((s: any) => ({
ID: s.id,
工卡有效期: s.card_expired,
……
}));

const sheet = XLSX.utils.json_to_sheet(data);
const wb = XLSX.utils.book_new();
XLSX.utils.book_append_sheet(wb, sheet, '員工信息表');
const wbbuf = XLSX.write(wb, {
  type: 'base64'
});

this.success({ name: "員工信息表.xlsx", data: wbbuf, type:xlsxMineType });

經過 json_to_sheet 能夠拿到包含單元格信息的對象

{ 
 A2: { t: 'n', v: 3776 },
 B2: { t: 's', v: '2019-04-01' },
 A3: { t: 'n', v: 3831 },
 B3: { t: 's', v: '2019-04-01' },
 A1: { t: 's', v: 'ID' },
 B1: { t: 's', v: '工卡有效期' },
 '!ref': 'A1:B3' 
}

對象中以單元格位置做爲key,每一個單元格的值(v)、類型(t)等等屬性做爲value。其中單元格的類型支持:

b Boolean, n Number, e error, s String, d Date

看起來Date類型十分符合上面的要求,就嘗試了一下:

const sheet = XLSX.utils.json_to_sheet(data);
// 篩選出除表頭的工卡列
Object.keys(sheet).filter(item => /^B/.test(item) && item !== "B1").forEach(key => {
   sheet[key].t = "d";
})

然鵝,若是工卡有效期原本就爲空,這時候導出,打開Excel會報錯,而且空的位置會變成NaN

翻閱了各類中英文文檔、Issue,導出一百多個員工信息表以後,我發現Excel真的很奇妙,或許應該在js上來格式化導入的數據,而不是限制單元格的類型。

若是不控制單元格類型的話,那麼當管理員輸入日期的時候,這個單元格多是:文本、常規、日期、自定義類型,因此只要保證無論單元格是什麼格式,程序都能拿到正確的數據就行了。

當管理員使的工卡有效期的單元格類型是文本或者常規的時候,則比較簡單,程序能夠按預期解析出來一個相應的字符串,用moment解析一下,就能夠得到想要的格式的數據了。

那麼當有效期單元格的類型是日期和自定義的時候,咱們拿到的數據是像下圖同樣

這也就是以前數據庫中奇怪的數字的由來,這個數字的意義,實際上是當前日期距離1900年1月0日的天數。還須要注意的是,Excel中有個bug:

它覺得1900年是閏年,因此咱們拿到的天數都會多了一天,由於轉換以前還須要先進行減一操做…

item.工卡有效期 = new Date(1900, 0, expried - 1)

這樣以後就能夠拿到正確的日期啦。咕嘰。

相關文章
相關標籤/搜索