thingsboard/ui-ngx/src/app/modules/home/components/widget/widget-component.service.ts

620 lines
25 KiB
TypeScript
Raw Normal View History

2019-09-06 20:17:45 +03:00
///
2022-01-17 14:07:46 +02:00
/// Copyright © 2016-2022 The Thingsboard Authors
2019-09-06 20:17:45 +03:00
///
/// Licensed under the Apache License, Version 2.0 (the "License");
/// you may not use this file except in compliance with the License.
/// You may obtain a copy of the License at
///
/// http://www.apache.org/licenses/LICENSE-2.0
///
/// Unless required by applicable law or agreed to in writing, software
/// distributed under the License is distributed on an "AS IS" BASIS,
/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
/// See the License for the specific language governing permissions and
/// limitations under the License.
///
import { ComponentFactory, Inject, Injectable, Optional, Type } from '@angular/core';
2019-09-06 20:17:45 +03:00
import { DynamicComponentFactoryService } from '@core/services/dynamic-component-factory.service';
import { WidgetService } from '@core/http/widget.service';
2020-12-28 16:06:36 +02:00
import { forkJoin, from, Observable, of, ReplaySubject, Subject, throwError } from 'rxjs';
import {
ErrorWidgetType,
MissingWidgetType,
toWidgetInfo,
toWidgetType,
WidgetInfo,
WidgetTypeInstance
} from '@home/models/widget-component.models';
2019-09-06 20:17:45 +03:00
import cssjs from '@core/css/css';
import { UtilsService } from '@core/services/utils.service';
import { ModulesWithFactories, ResourcesService } from '@core/services/resources.service';
2019-09-25 19:37:29 +03:00
import { Widget, widgetActionSources, WidgetControllerDescriptor, WidgetType } from '@shared/models/widget.models';
2020-12-28 16:06:36 +02:00
import { catchError, map, mergeMap, switchMap, tap } from 'rxjs/operators';
2019-09-06 20:17:45 +03:00
import { isFunction, isUndefined } from '@core/utils';
import { TranslateService } from '@ngx-translate/core';
import { DynamicWidgetComponent } from '@home/components/widget/dynamic-widget.component';
import { WidgetComponentsModule } from '@home/components/widget/widget-components.module';
2019-09-10 15:12:10 +03:00
import { WINDOW } from '@core/services/window.service';
import { NULL_UUID } from '@shared/models/id/has-uuid';
import { WidgetTypeId } from '@app/shared/models/id/widget-type-id';
import { TenantId } from '@app/shared/models/id/tenant-id';
2020-01-29 17:20:28 +02:00
import { SharedModule } from '@shared/shared.module';
2020-08-17 16:11:01 +03:00
import { MODULES_MAP } from '@shared/public-api';
2020-12-28 16:06:36 +02:00
import * as tinycolor_ from 'tinycolor2';
import moment from 'moment';
2021-12-06 12:54:48 +02:00
import { IModulesMap } from '@modules/common/modules-map.models';
import { HOME_COMPONENTS_MODULE_TOKEN } from '@home/components/tokens';
import { widgetSettingsComponentsMap } from '@home/components/widget/lib/settings/widget-settings.module';
2020-12-28 16:06:36 +02:00
const tinycolor = tinycolor_;
2019-09-10 15:12:10 +03:00
2019-12-23 14:36:44 +02:00
// @dynamic
2019-09-06 20:17:45 +03:00
@Injectable()
export class WidgetComponentService {
private cssParser = new cssjs();
private widgetsInfoInMemoryCache = new Map<string, WidgetInfo>();
private widgetsInfoFetchQueue = new Map<string, Array<Subject<WidgetInfo>>>();
private init$: Observable<any>;
private missingWidgetType: WidgetInfo;
private errorWidgetType: WidgetInfo;
private editingWidgetType: WidgetType;
2019-09-06 20:17:45 +03:00
2019-09-10 15:12:10 +03:00
constructor(@Inject(WINDOW) private window: Window,
2021-12-06 12:54:48 +02:00
@Optional() @Inject(MODULES_MAP) private modulesMap: IModulesMap,
@Inject(HOME_COMPONENTS_MODULE_TOKEN) private homeComponentsModule: Type<any>,
2019-09-10 15:12:10 +03:00
private dynamicComponentFactoryService: DynamicComponentFactoryService,
2019-09-06 20:17:45 +03:00
private widgetService: WidgetService,
private utils: UtilsService,
private resources: ResourcesService,
private translate: TranslateService) {
2019-09-10 15:12:10 +03:00
2019-09-06 20:17:45 +03:00
this.cssParser.testMode = false;
this.widgetService.onWidgetTypeUpdated().subscribe((widgetType) => {
this.deleteWidgetInfoFromCache(widgetType.bundleAlias, widgetType.alias, widgetType.tenantId.id === NULL_UUID);
});
this.widgetService.onWidgetBundleDeleted().subscribe((widgetsBundle) => {
this.deleteWidgetsBundleFromCache(widgetsBundle.alias, widgetsBundle.tenantId.id === NULL_UUID);
});
2019-09-06 20:17:45 +03:00
this.init();
}
private init(): Observable<any> {
if (this.init$) {
return this.init$;
} else {
this.missingWidgetType = {...MissingWidgetType};
this.errorWidgetType = {...ErrorWidgetType};
if (this.utils.widgetEditMode) {
this.editingWidgetType = toWidgetType(
{
widgetName: this.utils.editWidgetInfo.widgetName,
alias: 'customWidget',
type: this.utils.editWidgetInfo.type,
sizeX: this.utils.editWidgetInfo.sizeX,
sizeY: this.utils.editWidgetInfo.sizeY,
resources: this.utils.editWidgetInfo.resources,
templateHtml: this.utils.editWidgetInfo.templateHtml,
templateCss: this.utils.editWidgetInfo.templateCss,
controllerScript: this.utils.editWidgetInfo.controllerScript,
settingsSchema: this.utils.editWidgetInfo.settingsSchema,
dataKeySettingsSchema: this.utils.editWidgetInfo.dataKeySettingsSchema,
latestDataKeySettingsSchema: this.utils.editWidgetInfo.latestDataKeySettingsSchema,
settingsDirective: this.utils.editWidgetInfo.settingsDirective,
dataKeySettingsDirective: this.utils.editWidgetInfo.dataKeySettingsDirective,
2022-03-30 11:14:54 +03:00
latestDataKeySettingsDirective: this.utils.editWidgetInfo.latestDataKeySettingsDirective,
defaultConfig: this.utils.editWidgetInfo.defaultConfig
}, new WidgetTypeId('1'), new TenantId( NULL_UUID ), 'customWidgetBundle', undefined
);
}
2019-09-06 20:17:45 +03:00
const initSubject = new ReplaySubject();
this.init$ = initSubject.asObservable();
2020-12-28 16:06:36 +02:00
2021-01-05 11:37:05 +02:00
const w = (this.window as any);
w.tinycolor = tinycolor;
w.cssjs = cssjs;
w.moment = moment;
w.$ = $;
w.jQuery = $;
2020-12-28 16:06:36 +02:00
const widgetModulesTasks: Observable<any>[] = [];
2021-01-05 11:37:05 +02:00
widgetModulesTasks.push(from(import('jquery.terminal')));
widgetModulesTasks.push(from(import('flot/src/jquery.flot.js')).pipe(
mergeMap(() => {
const flotJsPluginsTasks: Observable<any>[] = [];
flotJsPluginsTasks.push(from(import('flot/lib/jquery.colorhelpers.js')));
flotJsPluginsTasks.push(from(import('flot/src/plugins/jquery.flot.time.js')));
flotJsPluginsTasks.push(from(import('flot/src/plugins/jquery.flot.selection.js')));
flotJsPluginsTasks.push(from(import('flot/src/plugins/jquery.flot.pie.js')));
flotJsPluginsTasks.push(from(import('flot/src/plugins/jquery.flot.crosshair.js')));
flotJsPluginsTasks.push(from(import('flot/src/plugins/jquery.flot.stack.js')));
flotJsPluginsTasks.push(from(import('flot/src/plugins/jquery.flot.symbol.js')));
flotJsPluginsTasks.push(from(import('flot.curvedlines/curvedLines.js')));
return forkJoin(flotJsPluginsTasks);
})
));
2020-12-28 16:06:36 +02:00
widgetModulesTasks.push(from(import('@home/components/widget/lib/flot-widget')).pipe(
tap((mod) => {
(window as any).TbFlot = mod.TbFlot;
}))
);
widgetModulesTasks.push(from(import('@home/components/widget/lib/analogue-compass')).pipe(
tap((mod) => {
(window as any).TbAnalogueCompass = mod.TbAnalogueCompass;
}))
);
widgetModulesTasks.push(from(import('@home/components/widget/lib/analogue-radial-gauge')).pipe(
tap((mod) => {
(window as any).TbAnalogueRadialGauge = mod.TbAnalogueRadialGauge;
}))
);
widgetModulesTasks.push(from(import('@home/components/widget/lib/analogue-linear-gauge')).pipe(
tap((mod) => {
(window as any).TbAnalogueLinearGauge = mod.TbAnalogueLinearGauge;
}))
);
widgetModulesTasks.push(from(import('@home/components/widget/lib/digital-gauge')).pipe(
tap((mod) => {
(window as any).TbCanvasDigitalGauge = mod.TbCanvasDigitalGauge;
}))
);
widgetModulesTasks.push(from(import('@home/components/widget/lib/maps/map-widget2')).pipe(
tap((mod) => {
(window as any).TbMapWidgetV2 = mod.TbMapWidgetV2;
}))
);
widgetModulesTasks.push(from(import('@home/components/widget/trip-animation/trip-animation.component')).pipe(
tap((mod) => {
(window as any).TbTripAnimationWidget = mod.TbTripAnimationWidget;
}))
);
forkJoin(widgetModulesTasks).subscribe(
2019-09-06 20:17:45 +03:00
() => {
2020-12-28 16:06:36 +02:00
const loadDefaultWidgetInfoTasks = [
this.loadWidgetResources(this.missingWidgetType, 'global-widget-missing-type',
[SharedModule, WidgetComponentsModule, this.homeComponentsModule]),
this.loadWidgetResources(this.errorWidgetType, 'global-widget-error-type',
[SharedModule, WidgetComponentsModule, this.homeComponentsModule]),
2020-12-28 16:06:36 +02:00
];
forkJoin(loadDefaultWidgetInfoTasks).subscribe(
() => {
initSubject.next();
},
(e) => {
let errorMessages = ['Failed to load default widget types!'];
if (e && e.length) {
errorMessages = errorMessages.concat(e);
}
console.error('Failed to load default widget types!');
initSubject.error({
widgetInfo: this.errorWidgetType,
errorMessages
});
}
);
2019-09-06 20:17:45 +03:00
},
2020-01-29 17:20:28 +02:00
(e) => {
2020-12-28 16:06:36 +02:00
let errorMessages = ['Failed to load widget modules!'];
2020-01-29 17:20:28 +02:00
if (e && e.length) {
errorMessages = errorMessages.concat(e);
}
2020-12-28 16:06:36 +02:00
console.error('Failed to load widget modules!');
2020-01-29 17:20:28 +02:00
initSubject.error({
widgetInfo: this.errorWidgetType,
errorMessages
});
2019-09-06 20:17:45 +03:00
}
);
return this.init$;
}
}
2019-09-25 19:37:29 +03:00
public getInstantWidgetInfo(widget: Widget): WidgetInfo {
const widgetInfo = this.getWidgetInfoFromCache(widget.bundleAlias, widget.typeAlias, widget.isSystemType);
if (widgetInfo) {
return widgetInfo;
} else {
return {} as WidgetInfo;
}
}
2019-09-06 20:17:45 +03:00
public getWidgetInfo(bundleAlias: string, widgetTypeAlias: string, isSystem: boolean): Observable<WidgetInfo> {
return this.init().pipe(
mergeMap(() => this.getWidgetInfoInternal(bundleAlias, widgetTypeAlias, isSystem))
);
}
private getWidgetInfoInternal(bundleAlias: string, widgetTypeAlias: string, isSystem: boolean): Observable<WidgetInfo> {
const widgetInfoSubject = new ReplaySubject<WidgetInfo>();
const widgetInfo = this.getWidgetInfoFromCache(bundleAlias, widgetTypeAlias, isSystem);
if (widgetInfo) {
widgetInfoSubject.next(widgetInfo);
widgetInfoSubject.complete();
} else {
if (this.utils.widgetEditMode) {
this.loadWidget(this.editingWidgetType, bundleAlias, isSystem, widgetInfoSubject);
2019-09-06 20:17:45 +03:00
} else {
const key = this.createWidgetInfoCacheKey(bundleAlias, widgetTypeAlias, isSystem);
let fetchQueue = this.widgetsInfoFetchQueue.get(key);
if (fetchQueue) {
fetchQueue.push(widgetInfoSubject);
} else {
fetchQueue = new Array<Subject<WidgetInfo>>();
this.widgetsInfoFetchQueue.set(key, fetchQueue);
this.widgetService.getWidgetType(bundleAlias, widgetTypeAlias, isSystem, {ignoreErrors: true}).subscribe(
2019-09-06 20:17:45 +03:00
(widgetType) => {
this.loadWidget(widgetType, bundleAlias, isSystem, widgetInfoSubject);
},
() => {
widgetInfoSubject.next(this.missingWidgetType);
widgetInfoSubject.complete();
this.resolveWidgetsInfoFetchQueue(key, this.missingWidgetType);
}
);
}
}
}
return widgetInfoSubject.asObservable();
}
private loadWidget(widgetType: WidgetType, bundleAlias: string, isSystem: boolean, widgetInfoSubject: Subject<WidgetInfo>) {
const widgetInfo = toWidgetInfo(widgetType);
const key = this.createWidgetInfoCacheKey(bundleAlias, widgetInfo.alias, isSystem);
let widgetControllerDescriptor: WidgetControllerDescriptor = null;
try {
widgetControllerDescriptor = this.createWidgetControllerDescriptor(widgetInfo, key);
} catch (e) {
const details = this.utils.parseException(e);
const errorMessage = `Failed to compile widget script. \n Error: ${details.message}`;
this.processWidgetLoadError([errorMessage], key, widgetInfoSubject);
}
if (widgetControllerDescriptor) {
const widgetNamespace = `widget-type-${(isSystem ? 'sys-' : '')}${bundleAlias}-${widgetInfo.alias}`;
this.loadWidgetResources(widgetInfo, widgetNamespace, [SharedModule, WidgetComponentsModule, this.homeComponentsModule]).subscribe(
2019-09-06 20:17:45 +03:00
() => {
if (widgetControllerDescriptor.settingsSchema) {
widgetInfo.typeSettingsSchema = widgetControllerDescriptor.settingsSchema;
}
if (widgetControllerDescriptor.dataKeySettingsSchema) {
widgetInfo.typeDataKeySettingsSchema = widgetControllerDescriptor.dataKeySettingsSchema;
}
if (widgetControllerDescriptor.latestDataKeySettingsSchema) {
widgetInfo.typeLatestDataKeySettingsSchema = widgetControllerDescriptor.latestDataKeySettingsSchema;
}
2019-09-06 20:17:45 +03:00
widgetInfo.typeParameters = widgetControllerDescriptor.typeParameters;
widgetInfo.actionSources = widgetControllerDescriptor.actionSources;
widgetInfo.widgetTypeFunction = widgetControllerDescriptor.widgetTypeFunction;
this.putWidgetInfoToCache(widgetInfo, bundleAlias, widgetInfo.alias, isSystem);
if (widgetInfoSubject) {
widgetInfoSubject.next(widgetInfo);
widgetInfoSubject.complete();
}
this.resolveWidgetsInfoFetchQueue(key, widgetInfo);
},
(errorMessages: string[]) => {
this.processWidgetLoadError(errorMessages, key, widgetInfoSubject);
}
);
}
}
2020-01-29 17:20:28 +02:00
private loadWidgetResources(widgetInfo: WidgetInfo, widgetNamespace: string, modules?: Type<any>[]): Observable<any> {
2019-09-06 20:17:45 +03:00
this.cssParser.cssPreviewNamespace = widgetNamespace;
this.cssParser.createStyleElement(widgetNamespace, widgetInfo.templateCss);
const resourceTasks: Observable<string>[] = [];
const modulesTasks: Observable<ModulesWithFactories | string>[] = [];
2019-09-06 20:17:45 +03:00
if (widgetInfo.resources.length > 0) {
widgetInfo.resources.filter(r => r.isModule).forEach(
(resource) => {
modulesTasks.push(
this.resources.loadFactories(resource.url, this.modulesMap).pipe(
catchError((e: Error) => of(e?.message ? e.message : `Failed to load widget resource module: '${resource.url}'`))
)
);
}
);
}
widgetInfo.resources.filter(r => !r.isModule).forEach(
(resource) => {
2019-09-06 20:17:45 +03:00
resourceTasks.push(
this.resources.loadResource(resource.url).pipe(
catchError(e => of(`Failed to load widget resource: '${resource.url}'`))
)
);
}
);
let modulesObservable: Observable<string | ModulesWithFactories>;
if (modulesTasks.length) {
modulesObservable = forkJoin(modulesTasks).pipe(
map(res => {
const msg = res.find(r => typeof r === 'string');
if (msg) {
return msg as string;
} else {
const modulesWithFactoriesList = res as ModulesWithFactories[];
const resModulesWithFactories: ModulesWithFactories = {
modules: modulesWithFactoriesList.map(mf => mf.modules).flat(),
factories: modulesWithFactoriesList.map(mf => mf.factories).flat()
};
if (modules && modules.length) {
resModulesWithFactories.modules.concat(modules);
}
return resModulesWithFactories;
}
})
);
} else {
modulesObservable = modules && modules.length ? of({modules, factories: []}) : of({modules: [], factories: []});
2019-09-06 20:17:45 +03:00
}
2019-09-06 20:17:45 +03:00
resourceTasks.push(
modulesObservable.pipe(
mergeMap((resolvedModules) => {
if (typeof resolvedModules === 'string') {
return of(resolvedModules);
} else {
this.registerWidgetSettingsForms(widgetInfo, resolvedModules.factories);
return this.dynamicComponentFactoryService.createDynamicComponentFactory(
class DynamicWidgetComponentInstance extends DynamicWidgetComponent {},
widgetInfo.templateHtml,
resolvedModules.modules
).pipe(
map((factory) => {
widgetInfo.componentFactory = factory;
return null;
}),
catchError(e => {
const details = this.utils.parseException(e);
const errorMessage = `Failed to compile widget html. \n Error: ${details.message}`;
return of(errorMessage);
})
2020-08-17 16:11:01 +03:00
);
}
}))
2019-09-06 20:17:45 +03:00
);
return forkJoin(resourceTasks).pipe(
switchMap(msgs => {
let errors: string[];
if (msgs && msgs.length) {
errors = msgs.filter(msg => msg && msg.length > 0);
}
if (errors && errors.length) {
return throwError(errors);
} else {
return of(null);
}
}
));
}
private registerWidgetSettingsForms(widgetInfo: WidgetInfo, factories: ComponentFactory<any>[]) {
const directives: string[] = [];
if (widgetInfo.settingsDirective && widgetInfo.settingsDirective.length) {
directives.push(widgetInfo.settingsDirective);
}
if (widgetInfo.dataKeySettingsDirective && widgetInfo.dataKeySettingsDirective.length) {
directives.push(widgetInfo.dataKeySettingsDirective);
}
if (widgetInfo.latestDataKeySettingsDirective && widgetInfo.latestDataKeySettingsDirective.length) {
directives.push(widgetInfo.latestDataKeySettingsDirective);
}
if (directives.length) {
factories.filter((factory) => directives.includes(factory.selector))
.forEach((foundFactory) => {
widgetSettingsComponentsMap[foundFactory.selector] = foundFactory.componentType;
});
}
}
2019-09-06 20:17:45 +03:00
private createWidgetControllerDescriptor(widgetInfo: WidgetInfo, name: string): WidgetControllerDescriptor {
let widgetTypeFunctionBody = `return function _${name} (ctx) {\n` +
2019-09-06 20:17:45 +03:00
' var self = this;\n' +
' self.ctx = ctx;\n\n'; /*+
' self.onInit = function() {\n\n' +
' }\n\n' +
' self.onDataUpdated = function() {\n\n' +
' }\n\n' +
' self.useCustomDatasources = function() {\n\n' +
' }\n\n' +
' self.typeParameters = function() {\n\n' +
return {
useCustomDatasources: false,
maxDatasources: -1, //unlimited
maxDataKeys: -1, //unlimited
dataKeysOptional: false,
stateData: false
};
' }\n\n' +
' self.actionSources = function() {\n\n' +
return {
'headerButton': {
name: 'Header button',
multiple: true
}
};
}\n\n' +
' self.onResize = function() {\n\n' +
' }\n\n' +
' self.onEditModeChanged = function() {\n\n' +
' }\n\n' +
' self.onMobileModeChanged = function() {\n\n' +
' }\n\n' +
' self.getSettingsSchema = function() {\n\n' +
' }\n\n' +
' self.getDataKeySettingsSchema = function() {\n\n' +
' }\n\n' +
' self.onDestroy = function() {\n\n' +
' }\n\n' +
'}';*/
widgetTypeFunctionBody += widgetInfo.controllerScript;
widgetTypeFunctionBody += '\n};\n';
try {
const widgetTypeFunction = new Function(widgetTypeFunctionBody);
const widgetType = widgetTypeFunction.apply(this);
const widgetTypeInstance: WidgetTypeInstance = new widgetType();
const result: WidgetControllerDescriptor = {
widgetTypeFunction: widgetType
};
if (isFunction(widgetTypeInstance.getSettingsSchema)) {
result.settingsSchema = widgetTypeInstance.getSettingsSchema();
}
if (isFunction(widgetTypeInstance.getDataKeySettingsSchema)) {
result.dataKeySettingsSchema = widgetTypeInstance.getDataKeySettingsSchema();
}
if (isFunction(widgetTypeInstance.getLatestDataKeySettingsSchema)) {
result.latestDataKeySettingsSchema = widgetTypeInstance.getLatestDataKeySettingsSchema();
}
2019-09-06 20:17:45 +03:00
if (isFunction(widgetTypeInstance.typeParameters)) {
result.typeParameters = widgetTypeInstance.typeParameters();
} else {
result.typeParameters = {};
}
if (isFunction(widgetTypeInstance.useCustomDatasources)) {
result.typeParameters.useCustomDatasources = widgetTypeInstance.useCustomDatasources();
} else {
result.typeParameters.useCustomDatasources = false;
}
2020-06-22 18:55:15 +03:00
if (isUndefined(result.typeParameters.hasDataPageLink)) {
result.typeParameters.hasDataPageLink = false;
}
2019-09-06 20:17:45 +03:00
if (isUndefined(result.typeParameters.maxDatasources)) {
result.typeParameters.maxDatasources = -1;
}
if (isUndefined(result.typeParameters.maxDataKeys)) {
result.typeParameters.maxDataKeys = -1;
}
2020-06-22 18:55:15 +03:00
if (isUndefined(result.typeParameters.singleEntity)) {
2020-06-25 20:08:07 +03:00
result.typeParameters.singleEntity = false;
2020-06-22 18:55:15 +03:00
}
if (isUndefined(result.typeParameters.hasAdditionalLatestDataKeys)) {
result.typeParameters.hasAdditionalLatestDataKeys = false;
}
if (isUndefined(result.typeParameters.warnOnPageDataOverflow)) {
result.typeParameters.warnOnPageDataOverflow = true;
}
2021-03-02 13:21:53 +02:00
if (isUndefined(result.typeParameters.ignoreDataUpdateOnIntervalTick)) {
result.typeParameters.ignoreDataUpdateOnIntervalTick = false;
}
2019-09-06 20:17:45 +03:00
if (isUndefined(result.typeParameters.dataKeysOptional)) {
result.typeParameters.dataKeysOptional = false;
}
if (isUndefined(result.typeParameters.datasourcesOptional)) {
result.typeParameters.datasourcesOptional = false;
}
2019-09-06 20:17:45 +03:00
if (isUndefined(result.typeParameters.stateData)) {
result.typeParameters.stateData = false;
}
if (isUndefined(result.typeParameters.processNoDataByWidget)) {
result.typeParameters.processNoDataByWidget = false;
}
2019-09-06 20:17:45 +03:00
if (isFunction(widgetTypeInstance.actionSources)) {
result.actionSources = widgetTypeInstance.actionSources();
} else {
result.actionSources = {};
}
for (const actionSourceId of Object.keys(widgetActionSources)) {
result.actionSources[actionSourceId] = {...widgetActionSources[actionSourceId]};
result.actionSources[actionSourceId].name = this.translate.instant(result.actionSources[actionSourceId].name);
}
return result;
} catch (e) {
this.utils.processWidgetException(e);
throw e;
}
}
private processWidgetLoadError(errorMessages: string[], cacheKey: string, widgetInfoSubject: Subject<WidgetInfo>) {
if (widgetInfoSubject) {
widgetInfoSubject.error({
widgetInfo: this.errorWidgetType,
errorMessages
});
}
this.resolveWidgetsInfoFetchQueue(cacheKey, this.errorWidgetType, errorMessages);
}
private resolveWidgetsInfoFetchQueue(key: string, widgetInfo: WidgetInfo, errorMessages?: string[]) {
const fetchQueue = this.widgetsInfoFetchQueue.get(key);
if (fetchQueue) {
fetchQueue.forEach(subject => {
if (!errorMessages) {
subject.next(widgetInfo);
subject.complete();
} else {
subject.error({
widgetInfo,
errorMessages
});
}
});
this.widgetsInfoFetchQueue.delete(key);
}
}
// Cache functions
private createWidgetInfoCacheKey(bundleAlias: string, widgetTypeAlias: string, isSystem: boolean): string {
return `${isSystem ? 'sys_' : ''}${bundleAlias}_${widgetTypeAlias}`;
}
private getWidgetInfoFromCache(bundleAlias: string, widgetTypeAlias: string, isSystem: boolean): WidgetInfo | undefined {
const key = this.createWidgetInfoCacheKey(bundleAlias, widgetTypeAlias, isSystem);
return this.widgetsInfoInMemoryCache.get(key);
}
private putWidgetInfoToCache(widgetInfo: WidgetInfo, bundleAlias: string, widgetTypeAlias: string, isSystem: boolean) {
const key = this.createWidgetInfoCacheKey(bundleAlias, widgetTypeAlias, isSystem);
this.widgetsInfoInMemoryCache.set(key, widgetInfo);
}
private deleteWidgetInfoFromCache(bundleAlias: string, widgetTypeAlias: string, isSystem: boolean) {
const key = this.createWidgetInfoCacheKey(bundleAlias, widgetTypeAlias, isSystem);
this.widgetsInfoInMemoryCache.delete(key);
}
private deleteWidgetsBundleFromCache(bundleAlias: string, isSystem: boolean) {
const key = (isSystem ? 'sys_' : '') + bundleAlias;
this.widgetsInfoInMemoryCache.forEach((widgetInfo, cacheKey) => {
if (cacheKey.startsWith(key)) {
this.widgetsInfoInMemoryCache.delete(cacheKey);
}
});
}
2019-09-06 20:17:45 +03:00
}