【问题标题】:Using external javascript libraries in angular 2 lazy loaded module, not index.html在 Angular 2 延迟加载模块中使用外部 javascript 库,而不是 index.html
【发布时间】:2017-03-04 07:52:07
【问题描述】:

所以我想在我的 Angular 2 应用程序中包含一个外部 JS 库,即 is case 中的轮播滑块。我看过很多教程,展示了如何添加它,我已经成功地做到了,但是引用 index.html 上的库。

如您所知,这将在每次访问应用程序时加载库,无论他们是否访问需要轮播的组件。由于它非常庞大,我只想将它加载到需要它的地方,即延迟加载模块中。

我还没有尝试过,但我确信我可以将脚本标签放入使用它的组件中,但这对我来说感觉不对。

必须有一个正确的方式。这是什么!?

谢谢!

【问题讨论】:

标签: angular angular-cli external-js


【解决方案1】:

我们正在我们的一个项目中这样做以动态加载 js 库

ScriptStore.ts 将包含本地或远程服务器上的脚本的 路径 和将用于动态加载脚本

 interface Scripts {
    name: string;
    src: string;
}  
export const ScriptStore: Scripts[] = [
    {name: 'filepicker', src: 'https://api.filestackapi.com/filestack.js'},
    { name: 'rangeSlider', src: '../../../assets/js/ion.rangeSlider.min.js' }
];

script.service.ts 是一个可注入服务,将处理脚本的加载,照原样复制script.service.ts

import {Injectable} from "@angular/core";
import {ScriptStore} from "./script.store";

declare var document: any;

@Injectable()
export class Script {

private scripts: any = {};

constructor() {
    ScriptStore.forEach((script: any) => {
        this.scripts[script.name] = {
            loaded: false,
            src: script.src
        };
    });
}

load(...scripts: string[]) {
    var promises: any[] = [];
    scripts.forEach((script) => promises.push(this.loadScript(script)));
    return Promise.all(promises);
}

loadScript(name: string) {
    return new Promise((resolve, reject) => {
        //resolve if already loaded
        if (this.scripts[name].loaded) {
            resolve({script: name, loaded: true, status: 'Already Loaded'});
        }
        else {
            //load script
            let script = document.createElement('script');
            script.type = 'text/javascript';
            script.src = this.scripts[name].src;
            if (script.readyState) {  //IE
                script.onreadystatechange = () => {
                    if (script.readyState === "loaded" || script.readyState === "complete") {
                        script.onreadystatechange = null;
                        this.scripts[name].loaded = true;
                        resolve({script: name, loaded: true, status: 'Loaded'});
                    }
                };
            } else {  //Others
                script.onload = () => {
                    this.scripts[name].loaded = true;
                    resolve({script: name, loaded: true, status: 'Loaded'});
                };
            }
            script.onerror = (error: any) => resolve({script: name, loaded: false, status: 'Loaded'});
            document.getElementsByTagName('head')[0].appendChild(script);
        }
    });
}

}

在需要的地方注入 ScriptService 并像这样加载 js 库

this.script.load('filepicker', 'rangeSlider').then(data => {
            console.log('script loaded ', data);
        }).catch(error => console.log(error));

【讨论】:

    【解决方案2】:

    这实际上取决于您尝试加载的库的类型以及您在应用程序上使用的模块加载器。

    我将假设您使用的是 Typescript 或至少是 ES6 语法。

    如果您的第 3 方依赖项是 AMD/UMD/CommonJS 模块

    在这种情况下,您可以在组件文件之上导入它。

    import {SomeClassName} from "path/to/library";
    

    如果您的第 3 方是全局模块(例如:JQuery 插件)

    在这种情况下,您只需要导入 js 文件,它将在全局变量中可用。

    import "path/to/library";
    

    结论

    在任何情况下,您都应该检查 3rd 方库是否已经有可用的类型,如果您使用的是 Typescript,这将对您有所帮助。

    根据您的模块加载器,您可能需要在此之前进行一些配置。但如果没有更多信息,很难为您提供帮助。

    最好, jpsfs

    【讨论】:

    • 如果我们在组件中使用最后一个 import 语句,它是否只能用于该组件?就像我必须按顺序使用 jquery、chart.js 和 mymade.js 文件一样,我可以这样使用吗:import "path/to/jquery"; import "path/to/chart.js"; import "path/to/mymade.js"???
    • @AnkurShah 没有。使用最后一个导入语句,它将全局加载文件。
    猜你喜欢
    • 2017-02-22
    • 2017-06-21
    • 1970-01-01
    • 2017-05-16
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2019-01-01
    相关资源
    最近更新 更多