@c8y/ngx-components
Version:
Angular modules for Cumulocity IoT applications
126 lines • 23.3 kB
JavaScript
import { Component, Input, Optional, ViewChild } from '@angular/core';
import { CommonModule as NgCommonModule } from '@angular/common';
import { FormsModule as NgFormsModule } from '@angular/forms';
import { InventoryService } from '@c8y/client';
import { ClusterMapComponent, MapModule } from '@c8y/ngx-components/map';
import { AssetLinkPipe, CommonModule, DashboardChildComponent, DismissAlertStrategy, DynamicComponentAlert, FormsModule, gettext } from '@c8y/ngx-components';
import { filter, takeUntil } from 'rxjs/operators';
import { Subject } from 'rxjs';
import { ContextDashboardComponent } from '@c8y/ngx-components/context-dashboard';
import { MapEventInfoComponent } from './map-event-info.component';
import { IconSelectorModule } from '@c8y/ngx-components/icon-selector';
import { TooltipModule } from 'ngx-bootstrap/tooltip';
import { RouterLink } from '@angular/router';
import * as i0 from "@angular/core";
import * as i1 from "@c8y/ngx-components";
import * as i2 from "@c8y/ngx-components/context-dashboard";
import * as i3 from "@c8y/client";
import * as i4 from "@angular/common";
import * as i5 from "@c8y/ngx-components/map";
export class MapWidgetComponent {
constructor(dashboardChild, dashboardContextComponent, inventory) {
this.dashboardContextComponent = dashboardContextComponent;
this.inventory = inventory;
this.mapConfig = { center: [0, 0] };
this.TIMEOUT_ERROR_TEXT = gettext('The request is taking longer than usual. We apologize for the inconvenience.');
this.SERVER_ERROR_TEXT = gettext('Server error occurred.');
this.destroy$ = new Subject();
this.listenToWidgetResizeEvent(dashboardChild);
}
async ngOnInit() {
this.alerts.setAlertGroupDismissStrategy('warning', DismissAlertStrategy.TEMPORARY_OR_PERMANENT);
if (this.dashboardContextComponent?.dashboard?.deviceType && !this.config.device) {
const context = this.dashboardContextComponent.context;
if (context?.id) {
const { id } = context;
this.config.device = (await this.inventory.detail(id)).data;
}
}
if (this.config.device) {
this.rootNode = this.config.device;
}
this.mapConfig = { ...this.config.mapConfig };
if (this.config.widgetInstanceGlobalTimeContext) {
this.updateMapConfigRealtime();
}
this.savedNode = this.rootNode;
}
ngOnChanges(changes) {
if (changes.config?.currentValue?.widgetInstanceGlobalTimeContext) {
this.updateMapConfigRealtime();
}
}
ngAfterViewInit() {
this.subscribeToErrorsOccurred();
}
startFollow(context) {
if (context.id !== this.rootNode?.id) {
this.rootNode = context;
}
this.mapConfig = {
...this.config.mapConfig,
follow: true,
realtime: true
};
}
stopFollow() {
this.mapConfig = {
...this.config.mapConfig,
follow: false
};
if (this.config.widgetInstanceGlobalTimeContext) {
this.updateMapConfigRealtime();
}
this.rootNode = this.savedNode;
}
ngOnDestroy() {
this.destroy$.next();
}
listenToWidgetResizeEvent(dashboardChild) {
dashboardChild.changeEnd
.pipe(filter(child => child.lastChange === 'resize'), takeUntil(this.destroy$))
.subscribe(() => {
this.clusterMap.reset();
});
}
subscribeToErrorsOccurred() {
this.clusterMap.errorNotifier
.pipe(filter(Boolean), takeUntil(this.destroy$))
.subscribe(error => {
this.alerts.addAlerts(new DynamicComponentAlert({
type: 'warning',
text: error?.name === 'TimeoutError'
? this.TIMEOUT_ERROR_TEXT
: error?.message ?? this.SERVER_ERROR_TEXT
}));
});
}
updateMapConfigRealtime() {
this.mapConfig.realtime = this.config.realtime || false;
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: MapWidgetComponent, deps: [{ token: i1.DashboardChildComponent }, { token: i2.ContextDashboardComponent, optional: true }, { token: i3.InventoryService }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: MapWidgetComponent, isStandalone: true, selector: "c8y-map-widget", inputs: { config: "config" }, viewQueries: [{ propertyName: "clusterMap", first: true, predicate: ClusterMapComponent, descendants: true }], usesOnChanges: true, ngImport: i0, template: "<c8y-map-status\n [clusterMap]=\"mapWidget\"\n [(config)]=\"mapConfig\"\n (onUnfollow)=\"stopFollow()\"\n [buttonsConfig]=\"config.widgetInstanceGlobalTimeContext ? { realtime: { show: false } } : null\"\n></c8y-map-status>\n<c8y-cluster-map\n #mapWidget\n [rootNode]=\"rootNode\"\n [config]=\"mapConfig\"\n>\n <div\n class=\"map-marker\"\n *c8yMapPopup=\"let context\"\n >\n <a\n class=\"text-truncate deviceLink text-12\"\n routerLink=\"{{ context | assetLink }}\"\n >\n <strong>{{ context.name }}</strong>\n </a>\n <c8y-map-event-info [asset]=\"context\">\n <button\n class=\"btn btn-default btn-xs btn-block m-t-8\"\n [title]=\"'Activate realtime on this asset and follow it if it moves' | translate\"\n type=\"button\"\n (click)=\"startFollow(context)\"\n *ngIf=\"!mapConfig.follow\"\n translate\n >\n Follow\n </button>\n <button\n class=\"btn btn-default btn-xs btn-block m-t-8\"\n [title]=\"'Stop following this asset.' | translate\"\n type=\"button\"\n (click)=\"stopFollow()\"\n *ngIf=\"mapConfig.follow\"\n translate\n >\n Unfollow\n </button>\n </c8y-map-event-info>\n </div>\n</c8y-cluster-map>\n", dependencies: [{ kind: "component", type: MapEventInfoComponent, selector: "c8y-map-event-info", inputs: ["asset"] }, { kind: "ngmodule", type: NgCommonModule }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: NgFormsModule }, { kind: "ngmodule", type: CommonModule }, { kind: "pipe", type: i1.C8yTranslatePipe, name: "translate" }, { kind: "directive", type: i1.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "pipe", type: i1.AssetLinkPipe, name: "assetLink" }, { kind: "ngmodule", type: FormsModule }, { kind: "ngmodule", type: IconSelectorModule }, { kind: "ngmodule", type: MapModule }, { kind: "component", type: i5.MapStatusComponent, selector: "c8y-map-status", inputs: ["config", "clusterMap", "buttonsConfig"], outputs: ["configChange", "onUnfollow"] }, { kind: "component", type: i5.ClusterMapComponent, selector: "c8y-cluster-map", inputs: ["config", "rootNode", "asset", "showClusterColor"], outputs: ["mapChange"] }, { kind: "directive", type: i5.MapPopupDirective, selector: "[c8yMapPopup]" }, { kind: "ngmodule", type: TooltipModule }, { kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }] }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: MapWidgetComponent, decorators: [{
type: Component,
args: [{ selector: 'c8y-map-widget', standalone: true, imports: [
MapEventInfoComponent,
NgCommonModule,
NgFormsModule,
CommonModule,
FormsModule,
IconSelectorModule,
MapModule,
AssetLinkPipe,
TooltipModule,
RouterLink
], template: "<c8y-map-status\n [clusterMap]=\"mapWidget\"\n [(config)]=\"mapConfig\"\n (onUnfollow)=\"stopFollow()\"\n [buttonsConfig]=\"config.widgetInstanceGlobalTimeContext ? { realtime: { show: false } } : null\"\n></c8y-map-status>\n<c8y-cluster-map\n #mapWidget\n [rootNode]=\"rootNode\"\n [config]=\"mapConfig\"\n>\n <div\n class=\"map-marker\"\n *c8yMapPopup=\"let context\"\n >\n <a\n class=\"text-truncate deviceLink text-12\"\n routerLink=\"{{ context | assetLink }}\"\n >\n <strong>{{ context.name }}</strong>\n </a>\n <c8y-map-event-info [asset]=\"context\">\n <button\n class=\"btn btn-default btn-xs btn-block m-t-8\"\n [title]=\"'Activate realtime on this asset and follow it if it moves' | translate\"\n type=\"button\"\n (click)=\"startFollow(context)\"\n *ngIf=\"!mapConfig.follow\"\n translate\n >\n Follow\n </button>\n <button\n class=\"btn btn-default btn-xs btn-block m-t-8\"\n [title]=\"'Stop following this asset.' | translate\"\n type=\"button\"\n (click)=\"stopFollow()\"\n *ngIf=\"mapConfig.follow\"\n translate\n >\n Unfollow\n </button>\n </c8y-map-event-info>\n </div>\n</c8y-cluster-map>\n" }]
}], ctorParameters: () => [{ type: i1.DashboardChildComponent }, { type: i2.ContextDashboardComponent, decorators: [{
type: Optional
}] }, { type: i3.InventoryService }], propDecorators: { config: [{
type: Input
}], clusterMap: [{
type: ViewChild,
args: [ClusterMapComponent]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"map-widget.component.js","sourceRoot":"","sources":["../../../../../widgets/implementations/map/map-widget.component.ts","../../../../../widgets/implementations/map/map-widget.component.html"],"names":[],"mappings":"AAAA,OAAO,EAEL,SAAS,EACT,KAAK,EAIL,QAAQ,EAER,SAAS,EACV,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,YAAY,IAAI,cAAc,EAAE,MAAM,iBAAiB,CAAC;AACjE,OAAO,EAAE,WAAW,IAAI,aAAa,EAAE,MAAM,gBAAgB,CAAC;AAC9D,OAAO,EAAkB,gBAAgB,EAAE,MAAM,aAAa,CAAC;AAC/D,OAAO,EAAE,mBAAmB,EAAoB,SAAS,EAAE,MAAM,yBAAyB,CAAC;AAC3F,OAAO,EACL,aAAa,EACb,YAAY,EACZ,uBAAuB,EACvB,oBAAoB,EAEpB,qBAAqB,EAErB,WAAW,EACX,OAAO,EACR,MAAM,qBAAqB,CAAC;AAE7B,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AACnD,OAAO,EAAE,OAAO,EAAE,MAAM,MAAM,CAAC;AAC/B,OAAO,EAAE,yBAAyB,EAAE,MAAM,uCAAuC,CAAC;AAClF,OAAO,EAAE,qBAAqB,EAAE,MAAM,4BAA4B,CAAC;AACnE,OAAO,EAAE,kBAAkB,EAAE,MAAM,mCAAmC,CAAC;AACvE,OAAO,EAAE,aAAa,EAAE,MAAM,uBAAuB,CAAC;AACtD,OAAO,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC;;;;;;;AAmB7C,MAAM,OAAO,kBAAkB;IAkB7B,YACE,cAAuC,EACnB,yBAAoD,EAChE,SAA2B;QADf,8BAAyB,GAAzB,yBAAyB,CAA2B;QAChE,cAAS,GAAT,SAAS,CAAkB;QAlBrC,cAAS,GAAqB,EAAE,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,CAAC;QAQzC,uBAAkB,GAAG,OAAO,CAClC,8EAA8E,CAC/E,CAAC;QACM,sBAAiB,GAAG,OAAO,CAAC,wBAAwB,CAAC,CAAC;QAEtD,aAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;QAOrC,IAAI,CAAC,yBAAyB,CAAC,cAAc,CAAC,CAAC;IACjD,CAAC;IAED,KAAK,CAAC,QAAQ;QACZ,IAAI,CAAC,MAAM,CAAC,4BAA4B,CACtC,SAAS,EACT,oBAAoB,CAAC,sBAAsB,CAC5C,CAAC;QACF,IAAI,IAAI,CAAC,yBAAyB,EAAE,SAAS,EAAE,UAAU,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC;YACjF,MAAM,OAAO,GAAG,IAAI,CAAC,yBAAyB,CAAC,OAAO,CAAC;YACvD,IAAI,OAAO,EAAE,EAAE,EAAE,CAAC;gBAChB,MAAM,EAAE,EAAE,EAAE,GAAG,OAAO,CAAC;gBACvB,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC,MAAM,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC;YAC9D,CAAC;QACH,CAAC;QACD,IAAI,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC;YACvB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC;QACrC,CAAC;QACD,IAAI,CAAC,SAAS,GAAG,EAAE,GAAG,IAAI,CAAC,MAAM,CAAC,SAAS,EAAE,CAAC;QAC9C,IAAI,IAAI,CAAC,MAAM,CAAC,+BAA+B,EAAE,CAAC;YAChD,IAAI,CAAC,uBAAuB,EAAE,CAAC;QACjC,CAAC;QACD,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,QAAQ,CAAC;IACjC,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,MAAM,EAAE,YAAY,EAAE,+BAA+B,EAAE,CAAC;YAClE,IAAI,CAAC,uBAAuB,EAAE,CAAC;QACjC,CAAC;IACH,CAAC;IAED,eAAe;QACb,IAAI,CAAC,yBAAyB,EAAE,CAAC;IACnC,CAAC;IAED,WAAW,CAAC,OAAO;QACjB,IAAI,OAAO,CAAC,EAAE,KAAK,IAAI,CAAC,QAAQ,EAAE,EAAE,EAAE,CAAC;YACrC,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC;QAC1B,CAAC;QACD,IAAI,CAAC,SAAS,GAAG;YACf,GAAG,IAAI,CAAC,MAAM,CAAC,SAAS;YACxB,MAAM,EAAE,IAAI;YACZ,QAAQ,EAAE,IAAI;SACf,CAAC;IACJ,CAAC;IAED,UAAU;QACR,IAAI,CAAC,SAAS,GAAG;YACf,GAAG,IAAI,CAAC,MAAM,CAAC,SAAS;YACxB,MAAM,EAAE,KAAK;SACd,CAAC;QACF,IAAI,IAAI,CAAC,MAAM,CAAC,+BAA+B,EAAE,CAAC;YAChD,IAAI,CAAC,uBAAuB,EAAE,CAAC;QACjC,CAAC;QACD,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC;IACjC,CAAC;IAED,WAAW;QACT,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;IACvB,CAAC;IAEO,yBAAyB,CAAC,cAAuC;QACvE,cAAc,CAAC,SAAS;aACrB,IAAI,CACH,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,UAAU,KAAK,QAAQ,CAAC,EAC9C,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CACzB;aACA,SAAS,CAAC,GAAG,EAAE;YACd,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;QAC1B,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,yBAAyB;QAC/B,IAAI,CAAC,UAAU,CAAC,aAAa;aAC1B,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;aAC/C,SAAS,CAAC,KAAK,CAAC,EAAE;YACjB,IAAI,CAAC,MAAM,CAAC,SAAS,CACnB,IAAI,qBAAqB,CAAC;gBACxB,IAAI,EAAE,SAAS;gBACf,IAAI,EACF,KAAK,EAAE,IAAI,KAAK,cAAc;oBAC5B,CAAC,CAAC,IAAI,CAAC,kBAAkB;oBACzB,CAAC,CAAC,KAAK,EAAE,OAAO,IAAI,IAAI,CAAC,iBAAiB;aAC/C,CAAC,CACH,CAAC;QACJ,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,uBAAuB;QAC7B,IAAI,CAAC,SAAS,CAAC,QAAQ,GAAG,IAAI,CAAC,MAAM,CAAC,QAAQ,IAAI,KAAK,CAAC;IAC1D,CAAC;+GAjHU,kBAAkB;mGAAlB,kBAAkB,oJAOlB,mBAAmB,qEC3DhC,qwCA6CA,4CDLI,qBAAqB,iFACrB,cAAc,kIACd,aAAa,8BACb,YAAY,2PACZ,WAAW,8BACX,kBAAkB,8BAClB,SAAS,+bAET,aAAa,+BACb,UAAU;;4FAGD,kBAAkB;kBAjB9B,SAAS;+BACE,gBAAgB,cAEd,IAAI,WACP;wBACP,qBAAqB;wBACrB,cAAc;wBACd,aAAa;wBACb,YAAY;wBACZ,WAAW;wBACX,kBAAkB;wBAClB,SAAS;wBACT,aAAa;wBACb,aAAa;wBACb,UAAU;qBACX;;0BAsBE,QAAQ;wEAfF,MAAM;sBAAd,KAAK;gBAGN,UAAU;sBADT,SAAS;uBAAC,mBAAmB","sourcesContent":["import {\n  AfterViewInit,\n  Component,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Optional,\n  SimpleChanges,\n  ViewChild\n} from '@angular/core';\nimport { CommonModule as NgCommonModule } from '@angular/common';\nimport { FormsModule as NgFormsModule } from '@angular/forms';\nimport { IManagedObject, InventoryService } from '@c8y/client';\nimport { ClusterMapComponent, ClusterMapConfig, MapModule } from '@c8y/ngx-components/map';\nimport {\n  AssetLinkPipe,\n  CommonModule,\n  DashboardChildComponent,\n  DismissAlertStrategy,\n  DynamicComponent,\n  DynamicComponentAlert,\n  DynamicComponentAlertAggregator,\n  FormsModule,\n  gettext\n} from '@c8y/ngx-components';\nimport { MapWidgetConfig } from './map-widget.model';\nimport { filter, takeUntil } from 'rxjs/operators';\nimport { Subject } from 'rxjs';\nimport { ContextDashboardComponent } from '@c8y/ngx-components/context-dashboard';\nimport { MapEventInfoComponent } from './map-event-info.component';\nimport { IconSelectorModule } from '@c8y/ngx-components/icon-selector';\nimport { TooltipModule } from 'ngx-bootstrap/tooltip';\nimport { RouterLink } from '@angular/router';\n\n@Component({\n  selector: 'c8y-map-widget',\n  templateUrl: './map-widget.component.html',\n  standalone: true,\n  imports: [\n    MapEventInfoComponent,\n    NgCommonModule,\n    NgFormsModule,\n    CommonModule,\n    FormsModule,\n    IconSelectorModule,\n    MapModule,\n    AssetLinkPipe,\n    TooltipModule,\n    RouterLink\n  ]\n})\nexport class MapWidgetComponent\n  implements OnInit, OnChanges, OnDestroy, AfterViewInit, DynamicComponent\n{\n  mapConfig: ClusterMapConfig = { center: [0, 0] };\n  rootNode: IManagedObject;\n  @Input() config: MapWidgetConfig;\n\n  @ViewChild(ClusterMapComponent)\n  clusterMap: ClusterMapComponent;\n  alerts: DynamicComponentAlertAggregator;\n\n  private TIMEOUT_ERROR_TEXT = gettext(\n    'The request is taking longer than usual. We apologize for the inconvenience.'\n  );\n  private SERVER_ERROR_TEXT = gettext('Server error occurred.');\n  private savedNode: IManagedObject;\n  private destroy$ = new Subject<void>();\n\n  constructor(\n    dashboardChild: DashboardChildComponent,\n    @Optional() private dashboardContextComponent: ContextDashboardComponent,\n    private inventory: InventoryService\n  ) {\n    this.listenToWidgetResizeEvent(dashboardChild);\n  }\n\n  async ngOnInit() {\n    this.alerts.setAlertGroupDismissStrategy(\n      'warning',\n      DismissAlertStrategy.TEMPORARY_OR_PERMANENT\n    );\n    if (this.dashboardContextComponent?.dashboard?.deviceType && !this.config.device) {\n      const context = this.dashboardContextComponent.context;\n      if (context?.id) {\n        const { id } = context;\n        this.config.device = (await this.inventory.detail(id)).data;\n      }\n    }\n    if (this.config.device) {\n      this.rootNode = this.config.device;\n    }\n    this.mapConfig = { ...this.config.mapConfig };\n    if (this.config.widgetInstanceGlobalTimeContext) {\n      this.updateMapConfigRealtime();\n    }\n    this.savedNode = this.rootNode;\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes.config?.currentValue?.widgetInstanceGlobalTimeContext) {\n      this.updateMapConfigRealtime();\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.subscribeToErrorsOccurred();\n  }\n\n  startFollow(context) {\n    if (context.id !== this.rootNode?.id) {\n      this.rootNode = context;\n    }\n    this.mapConfig = {\n      ...this.config.mapConfig,\n      follow: true,\n      realtime: true\n    };\n  }\n\n  stopFollow() {\n    this.mapConfig = {\n      ...this.config.mapConfig,\n      follow: false\n    };\n    if (this.config.widgetInstanceGlobalTimeContext) {\n      this.updateMapConfigRealtime();\n    }\n    this.rootNode = this.savedNode;\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n  }\n\n  private listenToWidgetResizeEvent(dashboardChild: DashboardChildComponent) {\n    dashboardChild.changeEnd\n      .pipe(\n        filter(child => child.lastChange === 'resize'),\n        takeUntil(this.destroy$)\n      )\n      .subscribe(() => {\n        this.clusterMap.reset();\n      });\n  }\n\n  private subscribeToErrorsOccurred() {\n    this.clusterMap.errorNotifier\n      .pipe(filter(Boolean), takeUntil(this.destroy$))\n      .subscribe(error => {\n        this.alerts.addAlerts(\n          new DynamicComponentAlert({\n            type: 'warning',\n            text:\n              error?.name === 'TimeoutError'\n                ? this.TIMEOUT_ERROR_TEXT\n                : error?.message ?? this.SERVER_ERROR_TEXT\n          })\n        );\n      });\n  }\n\n  private updateMapConfigRealtime() {\n    this.mapConfig.realtime = this.config.realtime || false;\n  }\n}\n","<c8y-map-status\n  [clusterMap]=\"mapWidget\"\n  [(config)]=\"mapConfig\"\n  (onUnfollow)=\"stopFollow()\"\n  [buttonsConfig]=\"config.widgetInstanceGlobalTimeContext ? { realtime: { show: false } } : null\"\n></c8y-map-status>\n<c8y-cluster-map\n  #mapWidget\n  [rootNode]=\"rootNode\"\n  [config]=\"mapConfig\"\n>\n  <div\n    class=\"map-marker\"\n    *c8yMapPopup=\"let context\"\n  >\n    <a\n      class=\"text-truncate deviceLink text-12\"\n      routerLink=\"{{ context | assetLink }}\"\n    >\n      <strong>{{ context.name }}</strong>\n    </a>\n    <c8y-map-event-info [asset]=\"context\">\n      <button\n        class=\"btn btn-default btn-xs btn-block m-t-8\"\n        [title]=\"'Activate realtime on this asset and follow it if it moves' | translate\"\n        type=\"button\"\n        (click)=\"startFollow(context)\"\n        *ngIf=\"!mapConfig.follow\"\n        translate\n      >\n        Follow\n      </button>\n      <button\n        class=\"btn btn-default btn-xs btn-block m-t-8\"\n        [title]=\"'Stop following this asset.' | translate\"\n        type=\"button\"\n        (click)=\"stopFollow()\"\n        *ngIf=\"mapConfig.follow\"\n        translate\n      >\n        Unfollow\n      </button>\n    </c8y-map-event-info>\n  </div>\n</c8y-cluster-map>\n"]}