Использование внешних библиотек JavaScript в угловых 2 ленивых загруженного модуля, а не index.html

Поэтому я хочу включить внешнюю библиотеку JS, ползунок карусели в моем случае, в мое приложение Angular 2. Я видел много учебников, которые показывают, как это можно добавить, я сделал это успешно, но со ссылкой на библиотеку в index.html.

Это, как вы знаете, будет загружать библиотеку при каждом посещении приложения, независимо от того, посещают ли они компоненты, нуждающиеся в карусели. Поскольку он довольно здоровенный, я хочу загружать его только там, где это необходимо, что находится внутри лениво загруженного модуля.

Я не пробовал, но я уверен, что мог бы просто добавить тег script в компонент, который его использует, но мне это не кажется правильным.

Должен быть правильный путь. Что это!?

Спасибо!

2 ответа

Решение

Мы делаем это в одном из наших проектов для динамической загрузки библиотек 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 libs вот так

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

Это действительно зависит от типа библиотеки, которую вы пытаетесь загрузить, и от того, какой загрузчик модулей вы используете в своем приложении.

Я предполагаю, что вы используете Typescript или, по крайней мере, синтаксис ES6.

Если ваша сторонняя зависимость - это модуль AMD/UMD/CommonJS

В этом случае поверх файла компонента вы можете просто импортировать его.

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

Если ваша третья сторона является глобальным модулем (например, плагин JQuery)

В этом случае вам просто нужно импортировать файл js, и он будет доступен для глобальной переменной.

import "path/to/library";

Заключение

В любом случае вам следует проверить, есть ли в сторонней библиотеке уже доступные наборы, что поможет вам, если вы используете Typescript.

В зависимости от вашего загрузчика модулей, вам может потребоваться выполнить некоторые настройки до этого. Но без дополнительной информации вам сложно помочь.

Лучший, JPSFS

Другие вопросы по тегам