如何在 Angular 中异步加载 momentjs 时区数据

Lie*_*ero 5 npm momentjs ecmascript-6 angular moment-timezone

我刚刚安装了这个包:

npm install moment-timezone --save
Run Code Online (Sandbox Code Playgroud)

在角度组件中,我像这样使用它:

import * as moment from 'moment-timezone';
moment().tz('America/New York');
Run Code Online (Sandbox Code Playgroud)

我想这会将所有时区数据 (900+kB) 添加到供应商包中并减慢应用程序启动速度。

如何仅按需异步加载?

我真的很想使用 NPM 解决方案,以便在更新 npm 包时自动更新数据库。没有人会记得手动更新时区数据库。

编辑:

@Dimanoid 的回答显示了如何在没有数据的情况下从 npm 导入时刻时区:

import * as moment from 'moment-timezone/moment-timezone';
Run Code Online (Sandbox Code Playgroud)

以下是将数据包含为延迟加载资产的方法:

  1. 修改angular.json -> projects -> yourprojects -> Architect -> build -> options:

    "assets": [
        "src/favicon.ico",
        "src/assets/i18n",
        {   //add this
            "glob": "latest.json",
            "input": "node_modules/moment-timezone/data/packed",
            "output": "assets/moment-timezone"
        }
    ]
    
    Run Code Online (Sandbox Code Playgroud)
  2. 从资产文件夹中按需请求:

    import * as moment from 'moment-timezone/moment-timezone';
    import { PlatformLocation } from '@angular/common';
    
    export class MyComponent implements OnInit {
    
      constructor(private platformLocation: PlatformLocation) {
      }
      ngOnInit(){
         //you can use @angular/http instead of fetch
         fetch(this.platformLocation.getBaseHrefFromDOM() +  'assets/moment-timezone/latest.json')
          .then(time => time.json())
          .then(timeZoneData => {
            moment.tz.load(timeZoneData);
          });
      }
    }
    
    Run Code Online (Sandbox Code Playgroud)

Dim*_*oid 2

好吧,为了不将评论转换为论坛,让我们总结一下这一切。

您可以从官方网站下载该库,将其放置在代码树中的某个位置,导入它,然后从后端加载 tz-bundle 并调用moment.tz.load()初始化moment-timezon然后使用它。

import 'moment';
import * as moment from '../../assets/moment-timezone-nodata';

ngOnInit() {
    this.loadTzBundle().subscribe(data => {
        moment.tz.load(data);
    });
}
Run Code Online (Sandbox Code Playgroud)

为了说明这一点,我做了一个最小的工作示例:https ://stackblitz.com/edit/moment-tz-data-loading

编辑

我查看了源代码,实际上 npm 包将库和数据分开了。它只是在导入默认值时加载数据。

var moment = module.exports = require("./moment-timezone");
moment.tz.load(require('./data/packed/latest.json'));
Run Code Online (Sandbox Code Playgroud)

因此,我认为您可以安装该软件包并加载没有数据的库。

import * as moment from 'moment-timezone/moment-timezone';
Run Code Online (Sandbox Code Playgroud)

我更新了这个例子。