2019-08-20 20:42:48 +03:00
|
|
|
///
|
|
|
|
|
/// Copyright © 2016-2019 The Thingsboard Authors
|
|
|
|
|
///
|
|
|
|
|
/// 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.
|
|
|
|
|
///
|
|
|
|
|
|
2019-09-06 20:17:45 +03:00
|
|
|
import { Injectable } from '@angular/core';
|
2020-02-04 15:14:17 +02:00
|
|
|
import { defaultHttpOptionsFromConfig, RequestConfig } from './http-utils';
|
|
|
|
|
import { Observable, of, ReplaySubject, Subject } from 'rxjs';
|
2019-09-06 20:17:45 +03:00
|
|
|
import { HttpClient } from '@angular/common/http';
|
|
|
|
|
import { PageLink } from '@shared/models/page/page-link';
|
|
|
|
|
import { PageData } from '@shared/models/page/page-data';
|
|
|
|
|
import { WidgetsBundle } from '@shared/models/widgets-bundle.model';
|
2020-02-04 15:14:17 +02:00
|
|
|
import { Widget, WidgetType, widgetType, widgetTypesData } from '@shared/models/widget.models';
|
2019-09-05 21:15:40 +03:00
|
|
|
import { UtilsService } from '@core/services/utils.service';
|
|
|
|
|
import { TranslateService } from '@ngx-translate/core';
|
|
|
|
|
import { ResourcesService } from '../services/resources.service';
|
2020-02-04 15:14:17 +02:00
|
|
|
import { toWidgetInfo, toWidgetType, WidgetInfo } from '@app/modules/home/models/widget-component.models';
|
|
|
|
|
import { filter, map, mergeMap, tap } from 'rxjs/operators';
|
2019-09-23 20:35:31 +03:00
|
|
|
import { WidgetTypeId } from '@shared/models/id/widget-type-id';
|
|
|
|
|
import { NULL_UUID } from '@shared/models/id/has-uuid';
|
|
|
|
|
import { ActivationEnd, Router } from '@angular/router';
|
2019-08-20 20:42:48 +03:00
|
|
|
|
|
|
|
|
@Injectable({
|
|
|
|
|
providedIn: 'root'
|
|
|
|
|
})
|
|
|
|
|
export class WidgetService {
|
|
|
|
|
|
2019-09-23 20:35:31 +03:00
|
|
|
private widgetTypeUpdatedSubject = new Subject<WidgetType>();
|
|
|
|
|
private widgetsBundleDeletedSubject = new Subject<WidgetsBundle>();
|
|
|
|
|
|
|
|
|
|
private allWidgetsBundles: Array<WidgetsBundle>;
|
|
|
|
|
private systemWidgetsBundles: Array<WidgetsBundle>;
|
|
|
|
|
private tenantWidgetsBundles: Array<WidgetsBundle>;
|
|
|
|
|
|
2019-08-20 20:42:48 +03:00
|
|
|
constructor(
|
2019-09-05 21:15:40 +03:00
|
|
|
private http: HttpClient,
|
|
|
|
|
private utils: UtilsService,
|
|
|
|
|
private resources: ResourcesService,
|
2019-09-23 20:35:31 +03:00
|
|
|
private translate: TranslateService,
|
|
|
|
|
private router: Router
|
2019-09-05 21:15:40 +03:00
|
|
|
) {
|
2019-09-23 20:35:31 +03:00
|
|
|
this.router.events.pipe(filter(event => event instanceof ActivationEnd)).subscribe(
|
|
|
|
|
() => {
|
|
|
|
|
this.invalidateWidgetsBundleCache();
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-15 12:22:14 +02:00
|
|
|
public getAllWidgetsBundles(config?: RequestConfig): Observable<Array<WidgetsBundle>> {
|
|
|
|
|
return this.loadWidgetsBundleCache(config).pipe(
|
2019-09-23 20:35:31 +03:00
|
|
|
map(() => this.allWidgetsBundles)
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-15 12:22:14 +02:00
|
|
|
public getSystemWidgetsBundles(config?: RequestConfig): Observable<Array<WidgetsBundle>> {
|
|
|
|
|
return this.loadWidgetsBundleCache(config).pipe(
|
2019-09-23 20:35:31 +03:00
|
|
|
map(() => this.systemWidgetsBundles)
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-15 12:22:14 +02:00
|
|
|
public getTenantWidgetsBundles(config?: RequestConfig): Observable<Array<WidgetsBundle>> {
|
|
|
|
|
return this.loadWidgetsBundleCache(config).pipe(
|
2019-09-23 20:35:31 +03:00
|
|
|
map(() => this.tenantWidgetsBundles)
|
|
|
|
|
);
|
2019-09-05 21:15:40 +03:00
|
|
|
}
|
2019-08-20 20:42:48 +03:00
|
|
|
|
2019-11-15 12:22:14 +02:00
|
|
|
public getWidgetBundles(pageLink: PageLink, config?: RequestConfig): Observable<PageData<WidgetsBundle>> {
|
2019-08-20 20:42:48 +03:00
|
|
|
return this.http.get<PageData<WidgetsBundle>>(`/api/widgetsBundles${pageLink.toQuery()}`,
|
2019-11-15 12:22:14 +02:00
|
|
|
defaultHttpOptionsFromConfig(config));
|
2019-08-20 20:42:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public getWidgetsBundle(widgetsBundleId: string,
|
2019-11-15 12:22:14 +02:00
|
|
|
config?: RequestConfig): Observable<WidgetsBundle> {
|
|
|
|
|
return this.http.get<WidgetsBundle>(`/api/widgetsBundle/${widgetsBundleId}`, defaultHttpOptionsFromConfig(config));
|
2019-08-20 20:42:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public saveWidgetsBundle(widgetsBundle: WidgetsBundle,
|
2019-11-15 12:22:14 +02:00
|
|
|
config?: RequestConfig): Observable<WidgetsBundle> {
|
2019-09-23 20:35:31 +03:00
|
|
|
return this.http.post<WidgetsBundle>('/api/widgetsBundle', widgetsBundle,
|
2019-11-15 12:22:14 +02:00
|
|
|
defaultHttpOptionsFromConfig(config)).pipe(
|
2019-09-23 20:35:31 +03:00
|
|
|
tap(() => {
|
|
|
|
|
this.invalidateWidgetsBundleCache();
|
|
|
|
|
})
|
|
|
|
|
);
|
2019-08-20 20:42:48 +03:00
|
|
|
}
|
|
|
|
|
|
2019-11-15 12:22:14 +02:00
|
|
|
public deleteWidgetsBundle(widgetsBundleId: string, config?: RequestConfig) {
|
|
|
|
|
return this.getWidgetsBundle(widgetsBundleId, config).pipe(
|
2019-09-23 20:35:31 +03:00
|
|
|
mergeMap((widgetsBundle) => {
|
|
|
|
|
return this.http.delete(`/api/widgetsBundle/${widgetsBundleId}`,
|
2019-11-15 12:22:14 +02:00
|
|
|
defaultHttpOptionsFromConfig(config)).pipe(
|
2019-09-23 20:35:31 +03:00
|
|
|
tap(() => {
|
|
|
|
|
this.invalidateWidgetsBundleCache();
|
|
|
|
|
this.widgetsBundleDeletedSubject.next(widgetsBundle);
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
));
|
2019-08-20 20:42:48 +03:00
|
|
|
}
|
|
|
|
|
|
2019-09-03 19:31:16 +03:00
|
|
|
public getBundleWidgetTypes(bundleAlias: string, isSystem: boolean,
|
2019-11-15 12:22:14 +02:00
|
|
|
config?: RequestConfig): Observable<Array<WidgetType>> {
|
2019-09-03 19:31:16 +03:00
|
|
|
return this.http.get<Array<WidgetType>>(`/api/widgetTypes?isSystem=${isSystem}&bundleAlias=${bundleAlias}`,
|
2019-11-15 12:22:14 +02:00
|
|
|
defaultHttpOptionsFromConfig(config));
|
2019-09-03 19:31:16 +03:00
|
|
|
}
|
|
|
|
|
|
2019-11-08 17:42:31 +02:00
|
|
|
public loadBundleLibraryWidgets(bundleAlias: string, isSystem: boolean,
|
2019-11-15 12:22:14 +02:00
|
|
|
config?: RequestConfig): Observable<Array<Widget>> {
|
|
|
|
|
return this.getBundleWidgetTypes(bundleAlias, isSystem, config).pipe(
|
2019-11-08 17:42:31 +02:00
|
|
|
map((types) => {
|
|
|
|
|
types = types.sort((a, b) => {
|
|
|
|
|
let result = widgetType[b.descriptor.type].localeCompare(widgetType[a.descriptor.type]);
|
|
|
|
|
if (result === 0) {
|
|
|
|
|
result = b.createdTime - a.createdTime;
|
|
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
});
|
|
|
|
|
const widgetTypes = new Array<Widget>();
|
|
|
|
|
let top = 0;
|
|
|
|
|
const lastTop = [0, 0, 0];
|
|
|
|
|
let col = 0;
|
|
|
|
|
let column = 0;
|
|
|
|
|
types.forEach((type) => {
|
|
|
|
|
const widgetTypeInfo = toWidgetInfo(type);
|
|
|
|
|
const sizeX = 8;
|
|
|
|
|
const sizeY = Math.floor(widgetTypeInfo.sizeY);
|
|
|
|
|
const widget: Widget = {
|
|
|
|
|
typeId: type.id,
|
|
|
|
|
isSystemType: isSystem,
|
|
|
|
|
bundleAlias,
|
|
|
|
|
typeAlias: widgetTypeInfo.alias,
|
|
|
|
|
type: widgetTypeInfo.type,
|
|
|
|
|
title: widgetTypeInfo.widgetName,
|
|
|
|
|
sizeX,
|
|
|
|
|
sizeY,
|
|
|
|
|
row: top,
|
|
|
|
|
col,
|
|
|
|
|
config: JSON.parse(widgetTypeInfo.defaultConfig)
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
widget.config.title = widgetTypeInfo.widgetName;
|
|
|
|
|
|
|
|
|
|
widgetTypes.push(widget);
|
|
|
|
|
top += sizeY;
|
|
|
|
|
if (top > lastTop[column] + 10) {
|
|
|
|
|
lastTop[column] = top;
|
|
|
|
|
column++;
|
|
|
|
|
if (column > 2) {
|
|
|
|
|
column = 0;
|
|
|
|
|
}
|
|
|
|
|
top = lastTop[column];
|
|
|
|
|
col = column * 8;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
return widgetTypes;
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2019-09-05 21:15:40 +03:00
|
|
|
public getWidgetType(bundleAlias: string, widgetTypeAlias: string, isSystem: boolean,
|
2019-11-15 12:22:14 +02:00
|
|
|
config?: RequestConfig): Observable<WidgetType> {
|
2019-09-05 21:15:40 +03:00
|
|
|
return this.http.get<WidgetType>(`/api/widgetType?isSystem=${isSystem}&bundleAlias=${bundleAlias}&alias=${widgetTypeAlias}`,
|
2019-11-15 12:22:14 +02:00
|
|
|
defaultHttpOptionsFromConfig(config));
|
2019-09-05 21:15:40 +03:00
|
|
|
}
|
2019-09-19 20:10:52 +03:00
|
|
|
|
2019-09-23 20:35:31 +03:00
|
|
|
public saveWidgetType(widgetInfo: WidgetInfo,
|
|
|
|
|
id: WidgetTypeId,
|
|
|
|
|
bundleAlias: string,
|
2019-11-15 12:22:14 +02:00
|
|
|
config?: RequestConfig): Observable<WidgetType> {
|
2019-09-23 20:35:31 +03:00
|
|
|
const widgetTypeInstance = toWidgetType(widgetInfo, id, undefined, bundleAlias);
|
|
|
|
|
return this.http.post<WidgetType>('/api/widgetType', widgetTypeInstance,
|
2019-11-15 12:22:14 +02:00
|
|
|
defaultHttpOptionsFromConfig(config)).pipe(
|
2019-09-23 20:35:31 +03:00
|
|
|
tap((savedWidgetType) => {
|
|
|
|
|
this.widgetTypeUpdatedSubject.next(savedWidgetType);
|
|
|
|
|
}));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public saveImportedWidgetType(widgetTypeInstance: WidgetType,
|
2019-11-15 12:22:14 +02:00
|
|
|
config?: RequestConfig): Observable<WidgetType> {
|
2019-09-23 20:35:31 +03:00
|
|
|
return this.http.post<WidgetType>('/api/widgetType', widgetTypeInstance,
|
2019-11-15 12:22:14 +02:00
|
|
|
defaultHttpOptionsFromConfig(config)).pipe(
|
2019-09-23 20:35:31 +03:00
|
|
|
tap((savedWidgetType) => {
|
|
|
|
|
this.widgetTypeUpdatedSubject.next(savedWidgetType);
|
|
|
|
|
}));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public deleteWidgetType(bundleAlias: string, widgetTypeAlias: string, isSystem: boolean,
|
2019-11-15 12:22:14 +02:00
|
|
|
config?: RequestConfig) {
|
|
|
|
|
return this.getWidgetType(bundleAlias, widgetTypeAlias, isSystem, config).pipe(
|
2019-09-23 20:35:31 +03:00
|
|
|
mergeMap((widgetTypeInstance) => {
|
|
|
|
|
return this.http.delete(`/api/widgetType/${widgetTypeInstance.id.id}`,
|
2019-11-15 12:22:14 +02:00
|
|
|
defaultHttpOptionsFromConfig(config)).pipe(
|
2019-09-23 20:35:31 +03:00
|
|
|
tap(() => {
|
|
|
|
|
this.widgetTypeUpdatedSubject.next(widgetTypeInstance);
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
));
|
|
|
|
|
}
|
|
|
|
|
|
2019-09-19 20:10:52 +03:00
|
|
|
public getWidgetTypeById(widgetTypeId: string,
|
2019-11-15 12:22:14 +02:00
|
|
|
config?: RequestConfig): Observable<WidgetType> {
|
2019-09-19 20:10:52 +03:00
|
|
|
return this.http.get<WidgetType>(`/api/widgetType/${widgetTypeId}`,
|
2019-11-15 12:22:14 +02:00
|
|
|
defaultHttpOptionsFromConfig(config));
|
2019-09-19 20:10:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public getWidgetTemplate(widgetTypeParam: widgetType,
|
2019-11-15 12:22:14 +02:00
|
|
|
config?: RequestConfig): Observable<WidgetInfo> {
|
2019-09-19 20:10:52 +03:00
|
|
|
const templateWidgetType = widgetTypesData.get(widgetTypeParam);
|
|
|
|
|
return this.getWidgetType(templateWidgetType.template.bundleAlias, templateWidgetType.template.alias, true,
|
2019-11-15 12:22:14 +02:00
|
|
|
config).pipe(
|
2019-09-19 20:10:52 +03:00
|
|
|
map((result) => {
|
|
|
|
|
const widgetInfo = toWidgetInfo(result);
|
|
|
|
|
widgetInfo.alias = undefined;
|
|
|
|
|
return widgetInfo;
|
|
|
|
|
})
|
|
|
|
|
);
|
2019-09-23 20:35:31 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public onWidgetTypeUpdated(): Observable<WidgetType> {
|
|
|
|
|
return this.widgetTypeUpdatedSubject.asObservable();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public onWidgetBundleDeleted(): Observable<WidgetsBundle> {
|
|
|
|
|
return this.widgetsBundleDeletedSubject.asObservable();
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-15 12:22:14 +02:00
|
|
|
private loadWidgetsBundleCache(config?: RequestConfig): Observable<any> {
|
2019-09-23 20:35:31 +03:00
|
|
|
if (!this.allWidgetsBundles) {
|
|
|
|
|
const loadWidgetsBundleCacheSubject = new ReplaySubject();
|
|
|
|
|
this.http.get<Array<WidgetsBundle>>('/api/widgetsBundles',
|
2019-11-15 12:22:14 +02:00
|
|
|
defaultHttpOptionsFromConfig(config)).subscribe(
|
2019-09-23 20:35:31 +03:00
|
|
|
(allWidgetsBundles) => {
|
|
|
|
|
this.allWidgetsBundles = allWidgetsBundles;
|
|
|
|
|
this.systemWidgetsBundles = new Array<WidgetsBundle>();
|
|
|
|
|
this.tenantWidgetsBundles = new Array<WidgetsBundle>();
|
|
|
|
|
this.allWidgetsBundles = this.allWidgetsBundles.sort((wb1, wb2) => {
|
|
|
|
|
let res = wb1.title.localeCompare(wb2.title);
|
|
|
|
|
if (res === 0) {
|
|
|
|
|
res = wb2.createdTime - wb1.createdTime;
|
|
|
|
|
}
|
|
|
|
|
return res;
|
|
|
|
|
});
|
|
|
|
|
this.allWidgetsBundles.forEach((widgetsBundle) => {
|
|
|
|
|
if (widgetsBundle.tenantId.id === NULL_UUID) {
|
|
|
|
|
this.systemWidgetsBundles.push(widgetsBundle);
|
|
|
|
|
} else {
|
|
|
|
|
this.tenantWidgetsBundles.push(widgetsBundle);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
loadWidgetsBundleCacheSubject.next();
|
|
|
|
|
loadWidgetsBundleCacheSubject.complete();
|
|
|
|
|
},
|
|
|
|
|
() => {
|
|
|
|
|
loadWidgetsBundleCacheSubject.error(null);
|
|
|
|
|
});
|
|
|
|
|
return loadWidgetsBundleCacheSubject.asObservable();
|
|
|
|
|
} else {
|
|
|
|
|
return of(null);
|
2019-09-19 20:10:52 +03:00
|
|
|
}
|
2019-09-23 20:35:31 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private invalidateWidgetsBundleCache() {
|
|
|
|
|
this.allWidgetsBundles = undefined;
|
|
|
|
|
this.systemWidgetsBundles = undefined;
|
|
|
|
|
this.tenantWidgetsBundles = undefined;
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-20 20:42:48 +03:00
|
|
|
}
|