Commit 9413b68ebf8ecae2fd6d2f79d06649d6bbb7e6ed

Authored by Mrkartoshka
1 parent 5596bd9f

Editable polygons

... ... @@ -6,19 +6,99 @@
6 6 },
7 7 "widgetTypes": [
8 8 {
9   - "alias": "markers_placement_openstreetmap",
10   - "name": "Markers Placement - OpenStreetMap",
  9 + "alias": "update_location_timeseries",
  10 + "name": "Update location timeseries",
11 11 "descriptor": {
12 12 "type": "latest",
13   - "sizeX": 8.5,
14   - "sizeY": 6.5,
  13 + "sizeX": 7.5,
  14 + "sizeY": 3,
15 15 "resources": [],
16   - "templateHtml": "",
17   - "templateCss": ".leaflet-zoom-box {\n\tz-index: 9;\n}\n\n.leaflet-pane { z-index: 4; }\n\n.leaflet-tile-pane { z-index: 2; }\n.leaflet-overlay-pane { z-index: 4; }\n.leaflet-shadow-pane { z-index: 5; }\n.leaflet-marker-pane { z-index: 6; }\n.leaflet-tooltip-pane { z-index: 7; }\n.leaflet-popup-pane { z-index: 8; }\n\n.leaflet-map-pane canvas { z-index: 1; }\n.leaflet-map-pane svg { z-index: 2; }\n\n.leaflet-control {\n\tz-index: 9;\n}\n.leaflet-top,\n.leaflet-bottom {\n\tz-index: 11;\n}\n\n.tb-marker-label {\n border: none;\n background: none;\n box-shadow: none;\n}\n\n.tb-marker-label:before {\n border: none;\n background: none;\n}\n",
18   - "controllerScript": "self.onInit = function() {\n self.ctx.map = new TbMapWidgetV2('openstreet-map', false, self.ctx, null, true);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbMapWidgetV2.settingsSchema('openstreet-map');\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbMapWidgetV2.dataKeySettingsSchema('openstreet-map');\n}\n\nself.actionSources = function() {\n return TbMapWidgetV2.actionSources();\n}\n\nself.onDestroy = function() {\n}\n\nself.typeParameters = function() {\n return {\n hasDataPageLink: true\n };\n}",
19   - "settingsSchema": "{}",
  16 + "templateHtml": "<div tb-toast toastTarget=\"{{ toastTargetId }}\" style=\"width: 100%; height: 100%;\">\n <form *ngIf=\"attributeUpdateFormGroup\"\n class=\"attribute-update-form\"\n [formGroup]=\"attributeUpdateFormGroup\"\n [ngClass]=\"{'small-width': smallWidthContainer}\"\n (ngSubmit)=\"updateAttribute()\">\n <div style=\"padding: 0 8px; margin: auto 0;\">\n <div class=\"attribute-update-form__grid\" [fxShow]=\"entityDetected && isValidParameter && dataKeyDetected\">\n <div class=\"grid__element\" \n [ngClass]=\"{'horizontal-alignment': isHorizontal && !changeAlignment}\">\n <mat-form-field class=\"mat-block\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? latLabel : '' }}</mat-label>\n <input matInput\n formControlName=\"currentLat\"\n required\n type=\"number\"\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n max=\"90\"\n min=\"-90\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentLat').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field>\n \n <mat-form-field class=\"mat-block\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? lngLabel : '' }}</mat-label>\n <input matInput\n formControlName=\"currentLng\"\n required\n type=\"number\"\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n max=\"180\"\n min=\"-180\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentLng').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field> \n </div>\n \n <div class=\"grid__element\">\n <button mat-button mat-icon-button class=\"getLocation\"\n type=\"button\"\n (click)=\"getCoordinate()\"\n *ngIf=\"settings.showGetLocation\"\n matTooltip=\"{{ 'widgets.input-widgets.get-location' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>my_location</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"applyChanges\"\n type=\"submit\"\n [disabled]=\"disableButton() || attributeUpdateFormGroup.invalid || attributeUpdateFormGroup.pristine\"\n matTooltip=\"{{ 'widgets.input-widgets.update-timeseries' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>check</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"discardChanges\"\n type=\"button\"\n [disabled]=\"disableButton()\"\n (click)=\"discardChange()\"\n matTooltip=\"{{ 'widgets.input-widgets.discard-changes' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>close</mat-icon>\n </button>\n </div>\n </div>\n \n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\" [fxHide]=\"entityDetected\" >\n {{ 'widgets.input-widgets.no-entity-selected' | translate }}\n </div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !dataKeyDetected\">\n {{ 'widgets.input-widgets.no-timeseries-selected' | translate }}\n </div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !isValidParameter\">\n {{ 'widgets.input-widgets.no-coordinate-specified' | translate }}\n </div>\n </div>\n </form>\n</div>",
  17 + "templateCss": ".attribute-update-form {\n overflow: hidden;\n height: 100%;\n display: flex;\n flex-direction: column;\n}\n\n.attribute-update-form__grid {\n display: flex;\n}\n.grid__element:first-child {\n flex-direction: column;\n flex: 1;\n}\n\n.grid__element.horizontal-alignment {\n flex-direction: row;\n}\n\n.grid__element:last-child {\n align-items: center;\n margin-left: 7px;\n}\n.grid__element {\n display: flex;\n}\n\n.attribute-update-form .mat-button.mat-icon-button {\n margin: 0;\n}\n\n.attribute-update-form .mat-button.mat-icon-button {\n width: 32px;\n min-width: 32px;\n height: 32px;\n min-height: 32px;\n padding: 0 !important;\n margin: 0;\n line-height: 20px;\n}\n\n.attribute-update-form .mat-button.getLocation {\n margin-right: 10px;\n}\n\n.attribute-update-form .mat-icon-button mat-icon {\n width: 20px;\n min-width: 20px;\n height: 20px;\n min-height: 20px;\n font-size: 20px;\n}\n\n.attribute-update-form mat-form-field{\n width: 100%;\n padding-right: 5px;\n}\n\n.attribute-update-form.small-width mat-form-field{\n width: 150px;\n}\n\n.tb-toast {\n font-size: 14px!important;\n}",
  18 + "controllerScript": "let $scope;\r\nlet settings;\r\nlet attributeService;\r\nlet utils;\r\nlet translate;\r\n\r\nself.onInit = function() {\r\n self.ctx.ngZone.run(function() {\r\n init(); \r\n self.ctx.detectChanges(true);\r\n });\r\n};\r\n\r\n\r\nfunction init() {\r\n $scope = self.ctx.$scope;\r\n attributeService = $scope.$injector.get(self.ctx.servicesMap.get('attributeService'));\r\n utils = $scope.$injector.get(self.ctx.servicesMap.get('utils'));\r\n translate = $scope.$injector.get(self.ctx.servicesMap.get('translate'));\r\n $scope.toastTargetId = 'input-widget' + utils.guid();\r\n settings = utils.deepClone(self.ctx.settings) || {};\r\n \r\n settings.showLabel = utils.defaultValue(settings.showLabel, true);\r\n settings.showResultMessage = utils.defaultValue(settings.showResultMessage, true);\r\n settings.showGetLocation = utils.defaultValue(settings.showGetLocation, true);\r\n settings.enableHighAccuracy = utils.defaultValue(settings.enableHighAccuracy, false);\r\n $scope.settings = settings;\r\n $scope.isValidParameter = true;\r\n $scope.dataKeyDetected = false; \r\n\r\n $scope.isHorizontal = (settings.inputFieldsAlignment === 'row');\r\n $scope.requiredErrorMessage = utils.customTranslation(settings.requiredErrorMessage, settings.requiredErrorMessage) || translate.instant('widgets.input-widgets.entity-coordinate-required');\r\n $scope.latLabel = utils.customTranslation(settings.latLabel, settings.latLabel) || translate.instant('widgets.input-widgets.latitude');\r\n $scope.lngLabel = utils.customTranslation(settings.lngLabel, settings.lngLabel) || translate.instant('widgets.input-widgets.longitude');\r\n\r\n $scope.attributeUpdateFormGroup = $scope.fb.group(\r\n {currentLat: [undefined, [$scope.validators.required,\r\n $scope.validators.min(-90),\r\n $scope.validators.max(90)]],\r\n currentLng: [undefined, [$scope.validators.required,\r\n $scope.validators.min(-180),\r\n $scope.validators.max(180)]]}\r\n );\r\n\r\n if (self.ctx.datasources && self.ctx.datasources.length) {\r\n var datasource = self.ctx.datasources[0];\r\n if (datasource.type === 'entity') {\r\n if (datasource.entityType && datasource.entityId) {\r\n $scope.entityName = datasource.entityName;\r\n if (settings.widgetTitle && settings.widgetTitle.length) {\r\n $scope.titleTemplate = utils.customTranslation(settings.widgetTitle, settings.widgetTitle);\r\n } else {\r\n $scope.titleTemplate = self.ctx.widgetConfig.title;\r\n }\r\n\r\n $scope.entityDetected = true;\r\n }\r\n }\r\n if (datasource.dataKeys.length > 1) {\r\n $scope.dataKeyDetected = true;\r\n for (let i = 0; i < datasource.dataKeys.length; i++) {\r\n if (datasource.dataKeys[i].type != \"timeseries\"){\r\n $scope.isValidParameter = false;\r\n }\r\n if (datasource.dataKeys[i].name !== settings.latKeyName && datasource.dataKeys[i].name !== settings.lngKeyName){\r\n $scope.dataKeyDetected = false;\r\n }\r\n }\r\n }\r\n }\r\n\r\n self.ctx.widgetTitle = utils.createLabelFromDatasource(self.ctx.datasources[0], $scope.titleTemplate);\r\n\r\n $scope.updateAttribute = function () {\r\n $scope.isFocused = false;\r\n if ($scope.entityDetected) {\r\n var datasource = self.ctx.datasources[0];\r\n\r\n attributeService.saveEntityTimeseries(\r\n datasource.entity.id,\r\n 'scope',\r\n [\r\n {\r\n key: settings.latKeyName,\r\n value: $scope.attributeUpdateFormGroup.get('currentLat').value\r\n },{\r\n key: settings.lngKeyName,\r\n value: $scope.attributeUpdateFormGroup.get('currentLng').value\r\n }\r\n ]\r\n ).subscribe(\r\n function success() {\r\n $scope.originalLat = $scope.attributeUpdateFormGroup.get('currentLat').value;\r\n $scope.originalLng = $scope.attributeUpdateFormGroup.get('currentLng').value;\r\n if (settings.showResultMessage) {\r\n $scope.showSuccessToast(translate.instant('widgets.input-widgets.update-successful'), 1000, 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n },\r\n function fail() {\r\n if (settings.showResultMessage) {\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.update-failed'), 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n }\r\n );\r\n }\r\n };\r\n\r\n $scope.changeFocus = function () {\r\n if ($scope.attributeUpdateFormGroup.get('currentLat').value === $scope.originalLat && $scope.attributeUpdateFormGroup.get('currentLng').value === $scope.originalLng) {\r\n $scope.isFocused = false;\r\n }\r\n };\r\n \r\n $scope.discardChange = function() {\r\n $scope.attributeUpdateFormGroup.setValue({\r\n 'currentLat': $scope.originalLat,\r\n 'currentLng': $scope.originalLng\r\n });\r\n $scope.isFocused = false;\r\n $scope.attributeUpdateFormGroup.markAsPristine();\r\n self.onDataUpdated();\r\n };\r\n \r\n $scope.disableButton = function () {\r\n return $scope.attributeUpdateFormGroup.get('currentLat').value === $scope.originalLat && $scope.attributeUpdateFormGroup.get('currentLng').value === $scope.originalLng || $scope.currentLng === null || $scope.currentLat === null;\r\n };\r\n \r\n $scope.getCoordinate = function() {\r\n if (navigator.geolocation) {\r\n navigator.geolocation.getCurrentPosition(showPosition, function (){\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.blocked-location'), \r\n 'bottom', 'left', $scope.toastTargetId);\r\n }, {\r\n enableHighAccuracy: settings.enableHighAccuracy\r\n });\r\n } else {\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.no-support-geolocation'), 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n };\r\n \r\n function showPosition(position) {\r\n $scope.attributeUpdateFormGroup.setValue({\r\n currentLat: correctValue(position.coords.latitude),\r\n currentLng: correctValue(position.coords.longitude)\r\n });\r\n $scope.attributeUpdateFormGroup.markAsDirty();\r\n $scope.isFocused = true;\r\n }\r\n \r\n self.onResize();\r\n}\r\n\r\nself.onDataUpdated = function() {\r\n try {\r\n if ($scope.dataKeyDetected) {\r\n if (!$scope.isFocused) {\r\n for(let i = 0; i < self.typeParameters().maxDataKeys; i++){\r\n if(self.ctx.data[i].dataKey.name === self.ctx.settings.latKeyName && $scope.attributeUpdateFormGroup.get('currentLat').pristine){\r\n $scope.originalLat = self.ctx.data[i].data[0][1];\r\n $scope.attributeUpdateFormGroup.get('currentLat').patchValue(correctValue($scope.originalLat));\r\n } else if(self.ctx.data[i].dataKey.name === self.ctx.settings.lngKeyName && $scope.attributeUpdateFormGroup.get('currentLng').pristine){\r\n $scope.originalLng = self.ctx.data[i].data[0][1];\r\n $scope.attributeUpdateFormGroup.get('currentLng').patchValue(correctValue($scope.originalLng));\r\n }\r\n }\r\n self.ctx.detectChanges();\r\n }\r\n }\r\n } catch (e) {\r\n console.log(e);\r\n }\r\n};\r\n\r\nfunction correctValue(value) {\r\n if (typeof value !== \"number\") {\r\n return 0;\r\n }\r\n return value;\r\n}\r\n\r\nself.onResize = function() {\r\n $scope.smallWidthContainer = (self.ctx.$container && self.ctx.$container[0].offsetWidth < 320);\r\n $scope.changeAlignment = ($scope.isHorizontal && self.ctx.$container && self.ctx.$container[0].offsetWidth < 480);\r\n self.ctx.detectChanges();\r\n};\r\n\r\nself.typeParameters = function() {\r\n return {\r\n maxDatasources: 1,\r\n maxDataKeys: 2,\r\n singleEntity: true\r\n };\r\n};\r\n\r\nself.onDestroy = function() {\r\n\r\n};",
  19 + "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"widgetTitle\": {\n \"title\": \"Widget title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"latKeyName\": {\n \"title\": \"Latitude key name\",\n \"type\": \"string\",\n \"default\": \"latitude\"\n },\n \"lngKeyName\": {\n \"title\": \"Longitude key name\",\n \"type\": \"string\",\n \"default\": \"longitude\"\n },\n \"showLabel\": {\n \"title\": \"Show label\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"latLabel\": {\n \"title\": \"Label for latitude\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"lngLabel\": {\n \"title\": \"Label for longitude\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"requiredErrorMessage\": {\n \"title\": \"'Required' error message\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"showResultMessage\": {\n \"title\": \"Show result message\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableHighAccuracy\": {\n \"title\": \"Use high accuracy\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"showGetLocation\": {\n \"title\": \"Show button 'Get current location'\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"inputFieldsAlignment\": {\n \"title\": \"Input fields alignment\",\n \"type\": \"string\",\n \"default\": \"column\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"widgetTitle\",\n \"latKeyName\",\n \"lngKeyName\",\n \"enableHighAccuracy\",\n \"showGetLocation\",\n \"showResultMessage\",\n {\n \"key\": \"inputFieldsAlignment\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"column\",\n \"label\": \"Column (default)\"\n },\n {\n \"value\": \"row\",\n \"label\": \"Row\"\n }\n ]\n },\n \"showLabel\",\n \"latLabel\",\n \"lngLabel\",\n \"requiredErrorMessage\"\n ]\n}",
  20 + "dataKeySettingsSchema": "{}\n",
  21 + "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Update location timeseries\",\"dropShadow\":true,\"enableFullscreen\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
  22 + }
  23 + },
  24 + {
  25 + "alias": "update_shared_double_attribute",
  26 + "name": "Update shared double attribute",
  27 + "descriptor": {
  28 + "type": "latest",
  29 + "sizeX": 7.5,
  30 + "sizeY": 3,
  31 + "resources": [],
  32 + "templateHtml": "<div tb-toast toastTarget=\"{{ toastTargetId }}\" style=\"width: 100%; height: 100%;\">\n <form *ngIf=\"attributeUpdateFormGroup\"\n class=\"attribute-update-form\"\n [formGroup]=\"attributeUpdateFormGroup\"\n (ngSubmit)=\"updateAttribute()\">\n <div style=\"padding: 0 8px; margin: auto 0;\">\n <div class=\"attribute-update-form__grid\" [fxShow]=\"entityDetected && isValidParameter && dataKeyDetected\">\n <div class=\"grid__element\">\n <mat-form-field class=\"mat-block\" style=\"width: 100%;\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? labelValue : '' }}</mat-label>\n <input matInput\n formControlName=\"currentValue\"\n required\n type=\"number\"\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n max=\"{{settings.maxValue}}\"\n min=\"{{settings.minValue}}\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentValue').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field> \n </div>\n \n <div class=\"grid__element\">\n <button mat-button mat-icon-button class=\"applyChanges\"\n type=\"submit\"\n [disabled]=\"originalValue === attributeUpdateFormGroup.get('currentValue').value || attributeUpdateFormGroup.invalid || !attributeUpdateFormGroup.dirty\"\n matTooltip=\"{{ 'widgets.input-widgets.update-attribute' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>check</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"discardChanges\"\n type=\"button\"\n [disabled]=\"originalValue === attributeUpdateFormGroup.get('currentValue').value\"\n (click)=\"attributeUpdateFormGroup.get('currentValue').patchValue(originalValue); isFocused = false\"\n matTooltip=\"{{ 'widgets.input-widgets.discard-changes' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>close</mat-icon>\n </button>\n </div>\n </div>\n \n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\" [fxHide]=\"entityDetected\" [innerHtml]=\"message\"></div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !dataKeyDetected\">\n {{ 'widgets.input-widgets.no-attribute-selected' | translate }}\n </div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !isValidParameter\">\n {{ 'widgets.input-widgets.timeseries-not-allowed' | translate }}\n </div>\n </div>\n </form>\n</div>",
  33 + "templateCss": ".attribute-update-form {\n overflow: hidden;\n height: 100%;\n display: flex;\n flex-direction: column;\n}\n\n.attribute-update-form__grid {\n display: flex;\n}\n.grid__element:first-child {\n flex: 1;\n}\n.grid__element:last-child {\n margin-top: 19px;\n margin-left: 7px;\n}\n.grid__element {\n display: flex;\n}\n\n.attribute-update-form .mat-button.mat-icon-button {\n width: 32px;\n min-width: 32px;\n height: 32px;\n min-height: 32px;\n padding: 0 !important;\n margin: 0 !important;\n line-height: 20px;\n}\n\n.attribute-update-form .mat-icon-button mat-icon {\n width: 20px;\n min-width: 20px;\n height: 20px;\n min-height: 20px;\n font-size: 20px;\n}\n\n.tb-toast {\n font-size: 14px!important;\n}",
  34 + "controllerScript": "let $scope;\nlet settings;\nlet attributeService;\nlet utils;\nlet translate;\n\nself.onInit = function() {\n self.ctx.ngZone.run(function() {\n init(); \n self.ctx.detectChanges(true);\n });\n};\n\n\nfunction init() {\n\n $scope = self.ctx.$scope;\n attributeService = $scope.$injector.get(self.ctx.servicesMap.get('attributeService'));\n utils = $scope.$injector.get(self.ctx.servicesMap.get('utils'));\n translate = $scope.$injector.get(self.ctx.servicesMap.get('translate'));\n $scope.toastTargetId = 'input-widget' + utils.guid();\n settings = utils.deepClone(self.ctx.settings) || {};\n settings.showLabel = utils.defaultValue(settings.showLabel, true);\n settings.showResultMessage = utils.defaultValue(settings.showResultMessage, true);\n $scope.settings = settings;\n $scope.isValidParameter = true;\n $scope.dataKeyDetected = false; \n $scope.message = translate.instant('widgets.input-widgets.no-entity-selected');\n \n $scope.requiredErrorMessage = utils.customTranslation(settings.requiredErrorMessage, settings.requiredErrorMessage) || translate.instant('widgets.input-widgets.entity-attribute-required');\n $scope.labelValue = utils.customTranslation(settings.labelValue, settings.labelValue) || translate.instant('widgets.input-widgets.value');\n\n $scope.attributeUpdateFormGroup = $scope.fb.group(\n {currentValue: [undefined, [$scope.validators.required,\n $scope.validators.min(settings.minValue),\n $scope.validators.max(settings.maxValue)]]}\n );\n\n if (self.ctx.datasources && self.ctx.datasources.length) {\n var datasource = self.ctx.datasources[0];\n if (datasource.type === 'entity') {\n if (datasource.entityType === 'DEVICE') {\n if (datasource.entityType && datasource.entityId) {\n $scope.entityName = datasource.entityName;\n if (settings.widgetTitle && settings.widgetTitle.length) {\n $scope.titleTemplate = utils.customTranslation(settings.widgetTitle, settings.widgetTitle);\n } else {\n $scope.titleTemplate = self.ctx.widgetConfig.title;\n }\n \n $scope.entityDetected = true;\n }\n } else {\n $scope.message = translate.instant('widgets.input-widgets.not-allowed-entity');\n }\n }\n if (datasource.dataKeys.length) {\n if (datasource.dataKeys[0].type !== \"attribute\") {\n $scope.isValidParameter = false;\n } else {\n $scope.currentKey = datasource.dataKeys[0].name;\n $scope.dataKeyType = datasource.dataKeys[0].type;\n $scope.dataKeyDetected = true;\n }\n }\n }\n\n self.ctx.widgetTitle = utils.createLabelFromDatasource(self.ctx.datasources[0], $scope.titleTemplate);\n\n $scope.updateAttribute = function () {\n $scope.isFocused = false;\n if ($scope.entityDetected) {\n var datasource = self.ctx.datasources[0];\n\n attributeService.saveEntityAttributes(\n datasource.entity.id,\n 'SHARED_SCOPE',\n [\n {\n key: $scope.currentKey,\n value: $scope.attributeUpdateFormGroup.get('currentValue').value\n }\n ]\n ).subscribe(\n function success() {\n $scope.originalValue = $scope.attributeUpdateFormGroup.get('currentValue').value;\n if (settings.showResultMessage) {\n $scope.showSuccessToast(translate.instant('widgets.input-widgets.update-successful'), 1000, 'bottom', 'left', $scope.toastTargetId);\n }\n },\n function fail() {\n if (settings.showResultMessage) {\n $scope.showErrorToast(translate.instant('widgets.input-widgets.update-failed'), 'bottom', 'left', $scope.toastTargetId);\n }\n }\n );\n }\n };\n\n $scope.changeFocus = function () {\n if ($scope.attributeUpdateFormGroup.get('currentValue').value === $scope.originalValue) {\n $scope.isFocused = false;\n }\n }\n}\n\nself.onDataUpdated = function() {\n\n try {\n if ($scope.dataKeyDetected) {\n if (!$scope.isFocused) {\n $scope.originalValue = self.ctx.data[0].data[0][1];\n $scope.attributeUpdateFormGroup.get('currentValue').patchValue(correctValue($scope.originalValue));\n self.ctx.detectChanges();\n }\n }\n } catch (e) {\n console.log(e);\n }\n}\n\nfunction correctValue(value) {\n if (typeof value !== \"number\") {\n return 0;\n }\n return value;\n}\n\nself.onResize = function() {\n\n}\n\nself.typeParameters = function() {\n return {\n maxDatasources: 1,\n maxDataKeys: 1,\n singleEntity: true\n }\n}\n\nself.onDestroy = function() {\n\n}",
  35 + "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"widgetTitle\": {\n \"title\": \"Widget title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"showLabel\":{\n \"title\":\"Show label\",\n \"type\":\"boolean\",\n \"default\":true\n },\n \"labelValue\": {\n \"title\": \"Label\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"requiredErrorMessage\": {\n \"title\": \"'Required' error message\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"maxValue\": {\n \"title\": \"Max value\",\n \"type\": \"number\",\n \"default\": \"\"\n },\n \"minValue\": {\n \"title\": \"Min value\",\n \"type\": \"number\",\n \"default\": \"\"\n },\n \"showResultMessage\":{\n \"title\":\"Show result message\",\n \"type\":\"boolean\",\n \"default\":true\n }\n },\n \"required\": []\n },\n \"form\": [\n \"widgetTitle\",\n \"showResultMessage\",\n \"showLabel\",\n \"labelValue\",\n \"requiredErrorMessage\",\n \"maxValue\",\n \"minValue\"\n ]\n}",
  36 + "dataKeySettingsSchema": "{}\n",
  37 + "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Update shared double attribute\",\"dropShadow\":true,\"enableFullscreen\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
  38 + }
  39 + },
  40 + {
  41 + "alias": "update_shared_integer_attribute",
  42 + "name": "Update shared integer attribute",
  43 + "descriptor": {
  44 + "type": "latest",
  45 + "sizeX": 7.5,
  46 + "sizeY": 3,
  47 + "resources": [],
  48 + "templateHtml": "<div tb-toast toastTarget=\"{{ toastTargetId }}\" style=\"width: 100%; height: 100%;\">\n <form *ngIf=\"attributeUpdateFormGroup\"\n class=\"attribute-update-form\"\n [formGroup]=\"attributeUpdateFormGroup\"\n (ngSubmit)=\"updateAttribute()\">\n <div style=\"padding: 0 8px; margin: auto 0;\">\n <div class=\"attribute-update-form__grid\" [fxShow]=\"entityDetected && isValidParameter && dataKeyDetected\">\n <div class=\"grid__element\">\n <mat-form-field class=\"mat-block\" style=\"width: 100%;\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? labelValue : '' }}</mat-label>\n <input matInput\n formControlName=\"currentValue\"\n required\n type=\"number\"\n step=\"1\"\n pattern=\"^-?[0-9]+$\"\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n max=\"{{settings.maxValue}}\"\n min=\"{{settings.minValue}}\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentValue').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field> \n </div>\n \n <div class=\"grid__element\">\n <button mat-button mat-icon-button class=\"applyChanges\"\n type=\"submit\"\n [disabled]=\"originalValue === attributeUpdateFormGroup.get('currentValue').value || attributeUpdateFormGroup.invalid || !attributeUpdateFormGroup.dirty\"\n matTooltip=\"{{ 'widgets.input-widgets.update-attribute' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>check</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"discardChanges\"\n type=\"button\"\n [disabled]=\"originalValue === attributeUpdateFormGroup.get('currentValue').value\"\n (click)=\"attributeUpdateFormGroup.get('currentValue').patchValue(originalValue); isFocused = false\"\n matTooltip=\"{{ 'widgets.input-widgets.discard-changes' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>close</mat-icon>\n </button>\n </div>\n </div>\n \n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\" [fxHide]=\"entityDetected\" [innerHtml]=\"message\"></div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !dataKeyDetected\">\n {{ 'widgets.input-widgets.no-attribute-selected' | translate }}\n </div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !isValidParameter\">\n {{ 'widgets.input-widgets.timeseries-not-allowed' | translate }}\n </div>\n </div>\n </form>\n</div>",
  49 + "templateCss": ".attribute-update-form {\n overflow: hidden;\n height: 100%;\n display: flex;\n flex-direction: column;\n}\n\n.attribute-update-form__grid {\n display: flex;\n}\n.grid__element:first-child {\n flex: 1;\n}\n.grid__element:last-child {\n margin-top: 19px;\n margin-left: 7px;\n}\n.grid__element {\n display: flex;\n}\n\n.attribute-update-form .mat-button.mat-icon-button {\n width: 32px;\n min-width: 32px;\n height: 32px;\n min-height: 32px;\n padding: 0 !important;\n margin: 0 !important;\n line-height: 20px;\n}\n\n.attribute-update-form .mat-icon-button mat-icon {\n width: 20px;\n min-width: 20px;\n height: 20px;\n min-height: 20px;\n font-size: 20px;\n}\n\n.tb-toast {\n font-size: 14px!important;\n}",
  50 + "controllerScript": "let $scope;\nlet settings;\nlet attributeService;\nlet utils;\nlet translate;\n\nself.onInit = function() {\n self.ctx.ngZone.run(function() {\n init(); \n self.ctx.detectChanges(true);\n });\n};\n\n\nfunction init() {\n\n $scope = self.ctx.$scope;\n attributeService = $scope.$injector.get(self.ctx.servicesMap.get('attributeService'));\n utils = $scope.$injector.get(self.ctx.servicesMap.get('utils'));\n translate = $scope.$injector.get(self.ctx.servicesMap.get('translate'));\n $scope.toastTargetId = 'input-widget' + utils.guid();\n settings = utils.deepClone(self.ctx.settings) || {};\n settings.showLabel = utils.defaultValue(settings.showLabel, true);\n settings.showResultMessage = utils.defaultValue(settings.showResultMessage, true);\n $scope.settings = settings;\n $scope.isValidParameter = true;\n $scope.dataKeyDetected = false; \n $scope.message = translate.instant('widgets.input-widgets.no-entity-selected');\n \n $scope.requiredErrorMessage = utils.customTranslation(settings.requiredErrorMessage, settings.requiredErrorMessage) || translate.instant('widgets.input-widgets.entity-attribute-required');\n $scope.labelValue = utils.customTranslation(settings.labelValue, settings.labelValue) || translate.instant('widgets.input-widgets.value');\n\n $scope.attributeUpdateFormGroup = $scope.fb.group(\n {currentValue: [undefined, [$scope.validators.required,\n $scope.validators.min(settings.minValue),\n $scope.validators.max(settings.maxValue),\n $scope.validators.pattern(/^-?[0-9]+$/)]]}\n );\n\n if (self.ctx.datasources && self.ctx.datasources.length) {\n var datasource = self.ctx.datasources[0];\n if (datasource.type === 'entity') {\n if (datasource.entityType === 'DEVICE') {\n if (datasource.entityType && datasource.entityId) {\n $scope.entityName = datasource.entityName;\n if (settings.widgetTitle && settings.widgetTitle.length) {\n $scope.titleTemplate = utils.customTranslation(settings.widgetTitle, settings.widgetTitle);\n } else {\n $scope.titleTemplate = self.ctx.widgetConfig.title;\n }\n \n $scope.entityDetected = true;\n }\n } else {\n $scope.message = translate.instant('widgets.input-widgets.not-allowed-entity');\n }\n }\n if (datasource.dataKeys.length) {\n if (datasource.dataKeys[0].type !== \"attribute\") {\n $scope.isValidParameter = false;\n } else {\n $scope.currentKey = datasource.dataKeys[0].name;\n $scope.dataKeyType = datasource.dataKeys[0].type;\n $scope.dataKeyDetected = true;\n }\n }\n }\n\n self.ctx.widgetTitle = utils.createLabelFromDatasource(self.ctx.datasources[0], $scope.titleTemplate);\n\n $scope.updateAttribute = function () {\n $scope.isFocused = false;\n if ($scope.entityDetected) {\n var datasource = self.ctx.datasources[0];\n\n attributeService.saveEntityAttributes(\n datasource.entity.id,\n 'SHARED_SCOPE',\n [\n {\n key: $scope.currentKey,\n value: $scope.attributeUpdateFormGroup.get('currentValue').value\n }\n ]\n ).subscribe(\n function success() {\n $scope.originalValue = $scope.attributeUpdateFormGroup.get('currentValue').value;\n if (settings.showResultMessage) {\n $scope.showSuccessToast(translate.instant('widgets.input-widgets.update-successful'), 1000, 'bottom', 'left', $scope.toastTargetId);\n }\n },\n function fail() {\n if (settings.showResultMessage) {\n $scope.showErrorToast(translate.instant('widgets.input-widgets.update-failed'), 'bottom', 'left', $scope.toastTargetId);\n }\n }\n );\n }\n };\n\n $scope.changeFocus = function () {\n if ($scope.attributeUpdateFormGroup.get('currentValue').value === $scope.originalValue) {\n $scope.isFocused = false;\n }\n }\n}\n\nself.onDataUpdated = function() {\n\n try {\n if ($scope.dataKeyDetected) {\n if (!$scope.isFocused) {\n $scope.originalValue = self.ctx.data[0].data[0][1];\n $scope.attributeUpdateFormGroup.get('currentValue').patchValue(correctValue($scope.originalValue));\n self.ctx.detectChanges();\n }\n }\n } catch (e) {\n console.log(e);\n }\n}\n\nfunction correctValue(value) {\n if (typeof value !== \"number\") {\n return 0;\n }\n return value;\n}\n\nself.onResize = function() {\n\n}\n\nself.typeParameters = function() {\n return {\n maxDatasources: 1,\n maxDataKeys: 1,\n singleEntity: true\n }\n}\n\nself.onDestroy = function() {\n\n}",
  51 + "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"widgetTitle\": {\n \"title\": \"Widget title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"showLabel\":{\n \"title\":\"Show label\",\n \"type\":\"boolean\",\n \"default\":true\n },\n \"labelValue\": {\n \"title\": \"Label\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"requiredErrorMessage\": {\n \"title\": \"'Required' error message\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"maxValue\": {\n \"title\": \"Max value\",\n \"type\": \"number\",\n \"default\": \"\"\n },\n \"minValue\": {\n \"title\": \"Min value\",\n \"type\": \"number\",\n \"default\": \"\"\n },\n \"showResultMessage\":{\n \"title\":\"Show result message\",\n \"type\":\"boolean\",\n \"default\":true\n }\n },\n \"required\": []\n },\n \"form\": [\n \"widgetTitle\",\n \"showResultMessage\",\n \"showLabel\",\n \"labelValue\",\n \"requiredErrorMessage\",\n \"maxValue\",\n \"minValue\"\n ]\n}",
  52 + "dataKeySettingsSchema": "{}\n",
  53 + "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Update shared integer attribute\",\"dropShadow\":true,\"enableFullscreen\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
  54 + }
  55 + },
  56 + {
  57 + "alias": "update_shared_string_attribute",
  58 + "name": "Update shared string attribute",
  59 + "descriptor": {
  60 + "type": "latest",
  61 + "sizeX": 7.5,
  62 + "sizeY": 3,
  63 + "resources": [],
  64 + "templateHtml": "<div tb-toast toastTarget=\"{{ toastTargetId }}\" style=\"width: 100%; height: 100%;\">\n <form *ngIf=\"attributeUpdateFormGroup\"\n class=\"attribute-update-form\"\n [formGroup]=\"attributeUpdateFormGroup\"\n (ngSubmit)=\"updateAttribute()\">\n <div style=\"padding: 0 8px; margin: auto 0;\">\n <div class=\"attribute-update-form__grid\" [fxShow]=\"entityDetected && isValidParameter && dataKeyDetected\">\n <div class=\"grid__element\">\n <mat-form-field class=\"mat-block\" style=\"width: 100%;\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? labelValue : '' }}</mat-label>\n <input matInput\n formControlName=\"currentValue\"\n required\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n maxlength=\"{{settings.maxLength}}\"\n minlength=\"{{settings.minLength}}\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentValue').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field> \n </div>\n \n <div class=\"grid__element\">\n <button mat-button mat-icon-button class=\"applyChanges\"\n type=\"submit\"\n [disabled]=\"originalValue === attributeUpdateFormGroup.get('currentValue').value || attributeUpdateFormGroup.invalid || !attributeUpdateFormGroup.dirty\"\n matTooltip=\"{{ 'widgets.input-widgets.update-attribute' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>check</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"discardChanges\"\n type=\"button\"\n [disabled]=\"originalValue === attributeUpdateFormGroup.get('currentValue').value\"\n (click)=\"attributeUpdateFormGroup.get('currentValue').patchValue(originalValue); isFocused = false\"\n matTooltip=\"{{ 'widgets.input-widgets.discard-changes' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>close</mat-icon>\n </button>\n </div>\n </div>\n \n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\" [fxHide]=\"entityDetected\" [innerHtml]=\"message\"></div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !dataKeyDetected\">\n {{ 'widgets.input-widgets.no-attribute-selected' | translate }}\n </div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !isValidParameter\">\n {{ 'widgets.input-widgets.timeseries-not-allowed' | translate }}\n </div>\n </div>\n </form>\n</div>",
  65 + "templateCss": ".attribute-update-form {\n overflow: hidden;\n height: 100%;\n display: flex;\n flex-direction: column;\n}\n\n.attribute-update-form__grid {\n display: flex;\n}\n.grid__element:first-child {\n flex: 1;\n}\n.grid__element:last-child {\n margin-top: 19px;\n margin-left: 7px;\n}\n.grid__element {\n display: flex;\n}\n\n.attribute-update-form .mat-button.mat-icon-button {\n width: 32px;\n min-width: 32px;\n height: 32px;\n min-height: 32px;\n padding: 0 !important;\n margin: 0 !important;\n line-height: 20px;\n}\n\n.attribute-update-form .mat-icon-button mat-icon {\n width: 20px;\n min-width: 20px;\n height: 20px;\n min-height: 20px;\n font-size: 20px;\n}\n\n.tb-toast {\n font-size: 14px!important;\n}",
  66 + "controllerScript": "let $scope;\nlet settings;\nlet attributeService;\nlet utils;\nlet translate;\n\nself.onInit = function() {\n self.ctx.ngZone.run(function() {\n init(); \n self.ctx.detectChanges(true);\n });\n};\n\n\nfunction init() {\n\n $scope = self.ctx.$scope;\n attributeService = $scope.$injector.get(self.ctx.servicesMap.get('attributeService'));\n utils = $scope.$injector.get(self.ctx.servicesMap.get('utils'));\n translate = $scope.$injector.get(self.ctx.servicesMap.get('translate'));\n $scope.toastTargetId = 'input-widget' + utils.guid();\n settings = utils.deepClone(self.ctx.settings) || {};\n settings.showLabel = utils.defaultValue(settings.showLabel, true);\n settings.showResultMessage = utils.defaultValue(settings.showResultMessage, true);\n $scope.settings = settings;\n $scope.isValidParameter = true;\n $scope.dataKeyDetected = false; \n $scope.message = translate.instant('widgets.input-widgets.no-entity-selected');\n \n $scope.requiredErrorMessage = utils.customTranslation(settings.requiredErrorMessage, settings.requiredErrorMessage) || translate.instant('widgets.input-widgets.entity-attribute-required');\n $scope.labelValue = utils.customTranslation(settings.labelValue, settings.labelValue) || translate.instant('widgets.input-widgets.value');\n\n $scope.attributeUpdateFormGroup = $scope.fb.group(\n {currentValue: [undefined, [$scope.validators.required,\n $scope.validators.minLength(settings.minLength),\n $scope.validators.maxLength(settings.maxLength)]]}\n );\n\n if (self.ctx.datasources && self.ctx.datasources.length) {\n var datasource = self.ctx.datasources[0];\n if (datasource.type === 'entity') {\n if (datasource.entityType === 'DEVICE') {\n if (datasource.entityType && datasource.entityId) {\n $scope.entityName = datasource.entityName;\n if (settings.widgetTitle && settings.widgetTitle.length) {\n $scope.titleTemplate = utils.customTranslation(settings.widgetTitle, settings.widgetTitle);\n } else {\n $scope.titleTemplate = self.ctx.widgetConfig.title;\n }\n \n $scope.entityDetected = true;\n }\n } else {\n $scope.message = translate.instant('widgets.input-widgets.not-allowed-entity');\n }\n }\n if (datasource.dataKeys.length) {\n if (datasource.dataKeys[0].type !== \"attribute\") {\n $scope.isValidParameter = false;\n } else {\n $scope.currentKey = datasource.dataKeys[0].name;\n $scope.dataKeyType = datasource.dataKeys[0].type;\n $scope.dataKeyDetected = true;\n }\n }\n }\n\n self.ctx.widgetTitle = utils.createLabelFromDatasource(self.ctx.datasources[0], $scope.titleTemplate);\n\n $scope.updateAttribute = function () {\n $scope.isFocused = false;\n if ($scope.entityDetected) {\n var datasource = self.ctx.datasources[0];\n\n attributeService.saveEntityAttributes(\n datasource.entity.id,\n 'SHARED_SCOPE',\n [\n {\n key: $scope.currentKey,\n value: $scope.attributeUpdateFormGroup.get('currentValue').value\n }\n ]\n ).subscribe(\n function success() {\n $scope.originalValue = $scope.attributeUpdateFormGroup.get('currentValue').value;\n if (settings.showResultMessage) {\n $scope.showSuccessToast(translate.instant('widgets.input-widgets.update-successful'), 1000, 'bottom', 'left', $scope.toastTargetId);\n }\n },\n function fail() {\n if (settings.showResultMessage) {\n $scope.showErrorToast(translate.instant('widgets.input-widgets.update-failed'), 'bottom', 'left', $scope.toastTargetId);\n }\n }\n );\n }\n };\n\n $scope.changeFocus = function () {\n if ($scope.attributeUpdateFormGroup.get('currentValue').value === $scope.originalValue) {\n $scope.isFocused = false;\n }\n }\n}\n\nself.onDataUpdated = function() {\n try {\n if ($scope.dataKeyDetected) {\n if (!$scope.isFocused) {\n $scope.originalValue = self.ctx.data[0].data[0][1];\n $scope.attributeUpdateFormGroup.get('currentValue').patchValue($scope.originalValue);\n self.ctx.detectChanges();\n }\n }\n } catch (e) {\n console.log(e);\n }\n}\n\nself.onResize = function() {\n\n}\n\nself.typeParameters = function() {\n return {\n maxDatasources: 1,\n maxDataKeys: 1,\n singleEntity: true\n }\n}\n\nself.onDestroy = function() {\n\n}",
  67 + "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"widgetTitle\": {\n \"title\": \"Widget title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"showLabel\":{\n \"title\":\"Show label\",\n \"type\":\"boolean\",\n \"default\":true\n },\n \"labelValue\": {\n \"title\": \"Label\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"requiredErrorMessage\": {\n \"title\": \"'Required' error message\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"maxLength\": {\n \"title\": \"Max length\",\n \"type\": \"number\",\n \"default\": \"\"\n },\n \"minLength\": {\n \"title\": \"Min length\",\n \"type\": \"number\",\n \"default\": \"\"\n },\n \"showResultMessage\":{\n \"title\":\"Show result message\",\n \"type\":\"boolean\",\n \"default\":true\n }\n },\n \"required\": []\n },\n \"form\": [\n \"widgetTitle\",\n \"showResultMessage\",\n \"showLabel\",\n \"labelValue\",\n \"requiredErrorMessage\",\n \"maxLength\",\n \"minLength\"\n ]\n}",
  68 + "dataKeySettingsSchema": "{}\n",
  69 + "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Sin\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.23592248334107624,\"funcBody\":\"return Math.round(1000*Math.sin(time/5000));\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Update shared string attribute\",\"dropShadow\":true,\"enableFullscreen\":false,\"enableDataExport\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
  70 + }
  71 + },
  72 + {
  73 + "alias": "update_server_location_attribute",
  74 + "name": "Update server location attribute",
  75 + "descriptor": {
  76 + "type": "latest",
  77 + "sizeX": 7.5,
  78 + "sizeY": 3,
  79 + "resources": [],
  80 + "templateHtml": "<div tb-toast toastTarget=\"{{ toastTargetId }}\" style=\"width: 100%; height: 100%;\">\n <form *ngIf=\"attributeUpdateFormGroup\"\n class=\"attribute-update-form\"\n [formGroup]=\"attributeUpdateFormGroup\"\n [ngClass]=\"{'small-width': smallWidthContainer}\"\n (ngSubmit)=\"updateAttribute()\">\n <div style=\"padding: 0 8px; margin: auto 0;\">\n <div class=\"attribute-update-form__grid\" [fxShow]=\"entityDetected && isValidParameter && dataKeyDetected\">\n <div class=\"grid__element\" \n [ngClass]=\"{'horizontal-alignment': isHorizontal && !changeAlignment}\">\n <mat-form-field class=\"mat-block\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? latLabel : '' }}</mat-label>\n <input matInput\n formControlName=\"currentLat\"\n required\n type=\"number\"\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n max=\"90\"\n min=\"-90\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentLat').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field>\n \n <mat-form-field class=\"mat-block\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? lngLabel : '' }}</mat-label>\n <input matInput\n formControlName=\"currentLng\"\n required\n type=\"number\"\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n max=\"180\"\n min=\"-180\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentLng').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field> \n </div>\n \n <div class=\"grid__element\">\n <button mat-button mat-icon-button class=\"getLocation\"\n type=\"button\"\n (click)=\"getCoordinate()\"\n *ngIf=\"settings.showGetLocation\"\n matTooltip=\"{{ 'widgets.input-widgets.get-location' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>my_location</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"applyChanges\"\n type=\"submit\"\n [disabled]=\"disableButton() || attributeUpdateFormGroup.invalid || attributeUpdateFormGroup.pristine\"\n matTooltip=\"{{ 'widgets.input-widgets.update-attribute' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>check</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"discardChanges\"\n type=\"button\"\n [disabled]=\"disableButton()\"\n (click)=\"discardChange()\"\n matTooltip=\"{{ 'widgets.input-widgets.discard-changes' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>close</mat-icon>\n </button>\n </div>\n </div>\n \n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\" [fxHide]=\"entityDetected\" >\n {{ 'widgets.input-widgets.no-entity-selected' | translate }}\n </div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !dataKeyDetected\">\n {{ 'widgets.input-widgets.no-attribute-selected' | translate }}\n </div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !isValidParameter\">\n {{ 'widgets.input-widgets.no-coordinate-specified' | translate }}\n </div>\n </div>\n </form>\n</div>",
  81 + "templateCss": ".attribute-update-form {\n overflow: hidden;\n height: 100%;\n display: flex;\n flex-direction: column;\n}\n\n.attribute-update-form__grid {\n display: flex;\n}\n.grid__element:first-child {\n flex-direction: column;\n flex: 1;\n}\n\n.grid__element.horizontal-alignment {\n flex-direction: row;\n}\n\n.grid__element:last-child {\n align-items: center;\n margin-left: 7px;\n}\n.grid__element {\n display: flex;\n}\n\n.attribute-update-form .mat-button.mat-icon-button {\n width: 32px;\n min-width: 32px;\n height: 32px;\n min-height: 32px;\n padding: 0 !important;\n margin: 0;\n line-height: 20px;\n}\n\n.attribute-update-form .mat-button.getLocation {\n margin-right: 10px;\n}\n\n.attribute-update-form .mat-icon-button mat-icon {\n width: 20px;\n min-width: 20px;\n height: 20px;\n min-height: 20px;\n font-size: 20px;\n}\n\n.attribute-update-form mat-form-field{\n width: 100%;\n padding-right: 5px;\n}\n\n.attribute-update-form.small-width mat-form-field{\n width: 150px;\n}\n\n.tb-toast {\n font-size: 14px!important;\n}",
  82 + "controllerScript": "let $scope;\r\nlet settings;\r\nlet attributeService;\r\nlet utils;\r\nlet translate;\r\n\r\nself.onInit = function() {\r\n self.ctx.ngZone.run(function() {\r\n init(); \r\n self.ctx.detectChanges(true);\r\n });\r\n};\r\n\r\n\r\nfunction init() {\r\n $scope = self.ctx.$scope;\r\n attributeService = $scope.$injector.get(self.ctx.servicesMap.get('attributeService'));\r\n utils = $scope.$injector.get(self.ctx.servicesMap.get('utils'));\r\n translate = $scope.$injector.get(self.ctx.servicesMap.get('translate'));\r\n $scope.toastTargetId = 'input-widget' + utils.guid();\r\n settings = utils.deepClone(self.ctx.settings) || {};\r\n \r\n settings.showLabel = utils.defaultValue(settings.showLabel, true);\r\n settings.showResultMessage = utils.defaultValue(settings.showResultMessage, true);\r\n settings.showGetLocation = utils.defaultValue(settings.showGetLocation, true);\r\n settings.enableHighAccuracy = utils.defaultValue(settings.enableHighAccuracy, false);\r\n $scope.settings = settings;\r\n $scope.isValidParameter = true;\r\n $scope.dataKeyDetected = false; \r\n\r\n $scope.isHorizontal = (settings.inputFieldsAlignment === 'row');\r\n $scope.requiredErrorMessage = utils.customTranslation(settings.requiredErrorMessage, settings.requiredErrorMessage) || translate.instant('widgets.input-widgets.entity-coordinate-required');\r\n $scope.latLabel = utils.customTranslation(settings.latLabel, settings.latLabel) || translate.instant('widgets.input-widgets.latitude');\r\n $scope.lngLabel = utils.customTranslation(settings.lngLabel, settings.lngLabel) || translate.instant('widgets.input-widgets.longitude');\r\n\r\n $scope.attributeUpdateFormGroup = $scope.fb.group(\r\n {currentLat: [undefined, [$scope.validators.required,\r\n $scope.validators.min(-90),\r\n $scope.validators.max(90)]],\r\n currentLng: [undefined, [$scope.validators.required,\r\n $scope.validators.min(-180),\r\n $scope.validators.max(180)]]}\r\n );\r\n\r\n if (self.ctx.datasources && self.ctx.datasources.length) {\r\n var datasource = self.ctx.datasources[0];\r\n if (datasource.type === 'entity') {\r\n if (datasource.entityType && datasource.entityId) {\r\n $scope.entityName = datasource.entityName;\r\n if (settings.widgetTitle && settings.widgetTitle.length) {\r\n $scope.titleTemplate = utils.customTranslation(settings.widgetTitle, settings.widgetTitle);\r\n } else {\r\n $scope.titleTemplate = self.ctx.widgetConfig.title;\r\n }\r\n\r\n $scope.entityDetected = true;\r\n }\r\n }\r\n if (datasource.dataKeys.length > 1) {\r\n $scope.dataKeyDetected = true;\r\n for (let i = 0; i < datasource.dataKeys.length; i++) {\r\n if (datasource.dataKeys[i].type != \"attribute\"){\r\n $scope.isValidParameter = false;\r\n }\r\n if (datasource.dataKeys[i].name !== settings.latKeyName && datasource.dataKeys[i].name !== settings.lngKeyName){\r\n $scope.dataKeyDetected = false;\r\n }\r\n }\r\n }\r\n }\r\n\r\n self.ctx.widgetTitle = utils.createLabelFromDatasource(self.ctx.datasources[0], $scope.titleTemplate);\r\n\r\n $scope.updateAttribute = function () {\r\n $scope.isFocused = false;\r\n if ($scope.entityDetected) {\r\n var datasource = self.ctx.datasources[0];\r\n\r\n attributeService.saveEntityAttributes(\r\n datasource.entity.id,\r\n 'SERVER_SCOPE',\r\n [\r\n {\r\n key: settings.latKeyName,\r\n value: $scope.attributeUpdateFormGroup.get('currentLat').value\r\n },{\r\n key: settings.lngKeyName,\r\n value: $scope.attributeUpdateFormGroup.get('currentLng').value\r\n }\r\n ]\r\n ).subscribe(\r\n function success() {\r\n $scope.originalLat = $scope.attributeUpdateFormGroup.get('currentLat').value;\r\n $scope.originalLng = $scope.attributeUpdateFormGroup.get('currentLng').value;\r\n if (settings.showResultMessage) {\r\n $scope.showSuccessToast(translate.instant('widgets.input-widgets.update-successful'), 1000, 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n },\r\n function fail() {\r\n if (settings.showResultMessage) {\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.update-failed'), 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n }\r\n );\r\n }\r\n };\r\n\r\n $scope.changeFocus = function () {\r\n if ($scope.attributeUpdateFormGroup.get('currentLat').value === $scope.originalLat && $scope.attributeUpdateFormGroup.get('currentLng').value === $scope.originalLng) {\r\n $scope.isFocused = false;\r\n }\r\n };\r\n \r\n $scope.discardChange = function() {\r\n $scope.attributeUpdateFormGroup.setValue({\r\n 'currentLat': $scope.originalLat,\r\n 'currentLng': $scope.originalLng\r\n });\r\n $scope.isFocused = false;\r\n $scope.attributeUpdateFormGroup.markAsPristine();\r\n self.onDataUpdated();\r\n };\r\n \r\n $scope.disableButton = function () {\r\n return $scope.attributeUpdateFormGroup.get('currentLat').value === $scope.originalLat && $scope.attributeUpdateFormGroup.get('currentLng').value === $scope.originalLng || $scope.currentLng === null || $scope.currentLat === null;\r\n };\r\n \r\n $scope.getCoordinate = function() {\r\n if (navigator.geolocation) {\r\n navigator.geolocation.getCurrentPosition(showPosition, function (){\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.blocked-location'), \r\n 'bottom', 'left', $scope.toastTargetId);\r\n }, {\r\n enableHighAccuracy: settings.enableHighAccuracy\r\n });\r\n } else {\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.no-support-geolocation'), 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n };\r\n \r\n function showPosition(position) {\r\n $scope.attributeUpdateFormGroup.setValue({\r\n currentLat: correctValue(position.coords.latitude),\r\n currentLng: correctValue(position.coords.longitude)\r\n });\r\n $scope.attributeUpdateFormGroup.markAsDirty();\r\n $scope.isFocused = true;\r\n }\r\n \r\n self.onResize();\r\n}\r\n\r\nself.onDataUpdated = function() {\r\n try {\r\n if ($scope.dataKeyDetected) {\r\n if (!$scope.isFocused) {\r\n for(let i = 0; i < self.typeParameters().maxDataKeys; i++){\r\n if(self.ctx.data[i].dataKey.name === self.ctx.settings.latKeyName && $scope.attributeUpdateFormGroup.get('currentLat').pristine){\r\n $scope.originalLat = self.ctx.data[i].data[0][1];\r\n $scope.attributeUpdateFormGroup.get('currentLat').patchValue(correctValue($scope.originalLat));\r\n } else if(self.ctx.data[i].dataKey.name === self.ctx.settings.lngKeyName && $scope.attributeUpdateFormGroup.get('currentLng').pristine){\r\n $scope.originalLng = self.ctx.data[i].data[0][1];\r\n $scope.attributeUpdateFormGroup.get('currentLng').patchValue(correctValue($scope.originalLng));\r\n }\r\n }\r\n self.ctx.detectChanges();\r\n }\r\n }\r\n } catch (e) {\r\n console.log(e);\r\n }\r\n};\r\n\r\nfunction correctValue(value) {\r\n if (typeof value !== \"number\") {\r\n return 0;\r\n }\r\n return value;\r\n}\r\n\r\nself.onResize = function() {\r\n $scope.smallWidthContainer = (self.ctx.$container && self.ctx.$container[0].offsetWidth < 320);\r\n $scope.changeAlignment = ($scope.isHorizontal && self.ctx.$container && self.ctx.$container[0].offsetWidth < 480);\r\n self.ctx.detectChanges();\r\n};\r\n\r\nself.typeParameters = function() {\r\n return {\r\n maxDatasources: 1,\r\n maxDataKeys: 2,\r\n singleEntity: true\r\n };\r\n};\r\n\r\nself.onDestroy = function() {\r\n\r\n};",
  83 + "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"widgetTitle\": {\n \"title\": \"Widget title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"latKeyName\": {\n \"title\": \"Latitude key name\",\n \"type\": \"string\",\n \"default\": \"latitude\"\n },\n \"lngKeyName\": {\n \"title\": \"Longitude key name\",\n \"type\": \"string\",\n \"default\": \"longitude\"\n },\n \"showLabel\": {\n \"title\": \"Show label\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"latLabel\": {\n \"title\": \"Label for latitude\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"lngLabel\": {\n \"title\": \"Label for longitude\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"requiredErrorMessage\": {\n \"title\": \"'Required' error message\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"showResultMessage\": {\n \"title\": \"Show result message\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableHighAccuracy\": {\n \"title\": \"Use high accuracy\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"showGetLocation\": {\n \"title\": \"Show button 'Get current location'\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"inputFieldsAlignment\": {\n \"title\": \"Input fields alignment\",\n \"type\": \"string\",\n \"default\": \"column\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"widgetTitle\",\n \"latKeyName\",\n \"lngKeyName\",\n \"enableHighAccuracy\",\n \"showGetLocation\",\n \"showResultMessage\",\n {\n \"key\": \"inputFieldsAlignment\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"column\",\n \"label\": \"Column (default)\"\n },\n {\n \"value\": \"row\",\n \"label\": \"Row\"\n }\n ]\n },\n \"showLabel\",\n \"latLabel\",\n \"lngLabel\",\n \"requiredErrorMessage\"\n ]\n}",
  84 + "dataKeySettingsSchema": "{}\n",
  85 + "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Update server location attribute\",\"dropShadow\":true,\"enableFullscreen\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
  86 + }
  87 + },
  88 + {
  89 + "alias": "update_shared_location_attribute",
  90 + "name": "Update shared location attribute",
  91 + "descriptor": {
  92 + "type": "latest",
  93 + "sizeX": 7.5,
  94 + "sizeY": 3,
  95 + "resources": [],
  96 + "templateHtml": "<div tb-toast toastTarget=\"{{ toastTargetId }}\" style=\"width: 100%; height: 100%;\">\n <form *ngIf=\"attributeUpdateFormGroup\"\n class=\"attribute-update-form\"\n [formGroup]=\"attributeUpdateFormGroup\"\n [ngClass]=\"{'small-width': smallWidthContainer}\"\n (ngSubmit)=\"updateAttribute()\">\n <div style=\"padding: 0 8px; margin: auto 0;\">\n <div class=\"attribute-update-form__grid\" [fxShow]=\"entityDetected && isValidParameter && dataKeyDetected\">\n <div class=\"grid__element\" \n [ngClass]=\"{'horizontal-alignment': isHorizontal && !changeAlignment}\">\n <mat-form-field class=\"mat-block\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? latLabel : '' }}</mat-label>\n <input matInput\n formControlName=\"currentLat\"\n required\n type=\"number\"\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n max=\"90\"\n min=\"-90\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentLat').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field>\n \n <mat-form-field class=\"mat-block\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? lngLabel : '' }}</mat-label>\n <input matInput\n formControlName=\"currentLng\"\n required\n type=\"number\"\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n max=\"180\"\n min=\"-180\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentLng').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field> \n </div>\n \n <div class=\"grid__element\">\n <button mat-button mat-icon-button class=\"getLocation\"\n type=\"button\"\n (click)=\"getCoordinate()\"\n *ngIf=\"settings.showGetLocation\"\n matTooltip=\"{{ 'widgets.input-widgets.get-location' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>my_location</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"applyChanges\"\n type=\"submit\"\n [disabled]=\"disableButton() || attributeUpdateFormGroup.invalid || attributeUpdateFormGroup.pristine\"\n matTooltip=\"{{ 'widgets.input-widgets.update-attribute' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>check</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"discardChanges\"\n type=\"button\"\n [disabled]=\"disableButton()\"\n (click)=\"discardChange()\"\n matTooltip=\"{{ 'widgets.input-widgets.discard-changes' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>close</mat-icon>\n </button>\n </div>\n </div>\n \n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\" [fxHide]=\"entityDetected\" [innerHtml]=\"message\"></div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !dataKeyDetected\">\n {{ 'widgets.input-widgets.no-attribute-selected' | translate }}\n </div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !isValidParameter\">\n {{ 'widgets.input-widgets.no-coordinate-specified' | translate }}\n </div>\n </div>\n </form>\n</div>",
  97 + "templateCss": ".attribute-update-form {\n overflow: hidden;\n height: 100%;\n display: flex;\n flex-direction: column;\n}\n\n.attribute-update-form__grid {\n display: flex;\n}\n.grid__element:first-child {\n flex-direction: column;\n flex: 1;\n}\n\n.grid__element.horizontal-alignment {\n flex-direction: row;\n}\n\n.grid__element:last-child {\n align-items: center;\n margin-left: 7px;\n}\n.grid__element {\n display: flex;\n}\n\n.attribute-update-form .mat-button.mat-icon-button {\n margin: 0;\n}\n\n.attribute-update-form .mat-button.mat-icon-button {\n width: 32px;\n min-width: 32px;\n height: 32px;\n min-height: 32px;\n padding: 0 !important;\n margin: 0;\n line-height: 20px;\n}\n\n.attribute-update-form .mat-button.getLocation {\n margin-right: 10px;\n}\n\n.attribute-update-form .mat-icon-button mat-icon {\n width: 20px;\n min-width: 20px;\n height: 20px;\n min-height: 20px;\n font-size: 20px;\n}\n\n.attribute-update-form mat-form-field{\n width: 100%;\n padding-right: 5px;\n}\n\n.attribute-update-form.small-width mat-form-field{\n width: 150px;\n}\n\n.tb-toast {\n font-size: 14px!important;\n}",
  98 + "controllerScript": "let $scope;\r\nlet settings;\r\nlet attributeService;\r\nlet utils;\r\nlet translate;\r\n\r\nself.onInit = function() {\r\n self.ctx.ngZone.run(function() {\r\n init(); \r\n self.ctx.detectChanges(true);\r\n });\r\n};\r\n\r\n\r\nfunction init() {\r\n $scope = self.ctx.$scope;\r\n attributeService = $scope.$injector.get(self.ctx.servicesMap.get('attributeService'));\r\n utils = $scope.$injector.get(self.ctx.servicesMap.get('utils'));\r\n translate = $scope.$injector.get(self.ctx.servicesMap.get('translate'));\r\n $scope.toastTargetId = 'input-widget' + utils.guid();\r\n settings = utils.deepClone(self.ctx.settings) || {};\r\n \r\n settings.showLabel = utils.defaultValue(settings.showLabel, true);\r\n settings.showResultMessage = utils.defaultValue(settings.showResultMessage, true);\r\n settings.showGetLocation = utils.defaultValue(settings.showGetLocation, true);\r\n settings.enableHighAccuracy = utils.defaultValue(settings.enableHighAccuracy, false);\r\n $scope.settings = settings;\r\n $scope.isValidParameter = true;\r\n $scope.dataKeyDetected = false; \r\n $scope.message = translate.instant('widgets.input-widgets.no-entity-selected');\r\n\r\n $scope.isHorizontal = (settings.inputFieldsAlignment === 'row');\r\n $scope.requiredErrorMessage = utils.customTranslation(settings.requiredErrorMessage, settings.requiredErrorMessage) || translate.instant('widgets.input-widgets.entity-coordinate-required');\r\n $scope.latLabel = utils.customTranslation(settings.latLabel, settings.latLabel) || translate.instant('widgets.input-widgets.latitude');\r\n $scope.lngLabel = utils.customTranslation(settings.lngLabel, settings.lngLabel) || translate.instant('widgets.input-widgets.longitude');\r\n\r\n $scope.attributeUpdateFormGroup = $scope.fb.group(\r\n {currentLat: [undefined, [$scope.validators.required,\r\n $scope.validators.min(-90),\r\n $scope.validators.max(90)]],\r\n currentLng: [undefined, [$scope.validators.required,\r\n $scope.validators.min(-180),\r\n $scope.validators.max(180)]]}\r\n );\r\n\r\n if (self.ctx.datasources && self.ctx.datasources.length) {\r\n var datasource = self.ctx.datasources[0];\r\n if (datasource.type === 'entity') {\r\n if (datasource.entityType === 'DEVICE') {\r\n if (datasource.entityType && datasource.entityId) {\r\n $scope.entityName = datasource.entityName;\r\n if (settings.widgetTitle && settings.widgetTitle.length) {\r\n $scope.titleTemplate = utils.customTranslation(settings.widgetTitle, settings.widgetTitle);\r\n } else {\r\n $scope.titleTemplate = self.ctx.widgetConfig.title;\r\n }\r\n \r\n $scope.entityDetected = true;\r\n }\r\n } else {\r\n $scope.message = translate.instant('widgets.input-widgets.not-allowed-entity');\r\n }\r\n }\r\n if (datasource.dataKeys.length > 1) {\r\n $scope.dataKeyDetected = true;\r\n for (let i = 0; i < datasource.dataKeys.length; i++) {\r\n if (datasource.dataKeys[i].type != \"attribute\"){\r\n $scope.isValidParameter = false;\r\n }\r\n if (datasource.dataKeys[i].name !== settings.latKeyName && datasource.dataKeys[i].name !== settings.lngKeyName){\r\n $scope.dataKeyDetected = false;\r\n }\r\n }\r\n }\r\n }\r\n\r\n self.ctx.widgetTitle = utils.createLabelFromDatasource(self.ctx.datasources[0], $scope.titleTemplate);\r\n\r\n $scope.updateAttribute = function () {\r\n $scope.isFocused = false;\r\n if ($scope.entityDetected) {\r\n var datasource = self.ctx.datasources[0];\r\n\r\n attributeService.saveEntityAttributes(\r\n datasource.entity.id,\r\n 'SHARED_SCOPE',\r\n [\r\n {\r\n key: settings.latKeyName,\r\n value: $scope.attributeUpdateFormGroup.get('currentLat').value\r\n },{\r\n key: settings.lngKeyName,\r\n value: $scope.attributeUpdateFormGroup.get('currentLng').value\r\n }\r\n ]\r\n ).subscribe(\r\n function success() {\r\n $scope.originalLat = $scope.attributeUpdateFormGroup.get('currentLat').value;\r\n $scope.originalLng = $scope.attributeUpdateFormGroup.get('currentLng').value;\r\n if (settings.showResultMessage) {\r\n $scope.showSuccessToast(translate.instant('widgets.input-widgets.update-successful'), 1000, 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n },\r\n function fail() {\r\n if (settings.showResultMessage) {\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.update-failed'), 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n }\r\n );\r\n }\r\n };\r\n\r\n $scope.changeFocus = function () {\r\n if ($scope.attributeUpdateFormGroup.get('currentLat').value === $scope.originalLat && $scope.attributeUpdateFormGroup.get('currentLng').value === $scope.originalLng) {\r\n $scope.isFocused = false;\r\n }\r\n };\r\n \r\n $scope.discardChange = function() {\r\n $scope.attributeUpdateFormGroup.setValue({\r\n 'currentLat': $scope.originalLat,\r\n 'currentLng': $scope.originalLng\r\n });\r\n $scope.isFocused = false;\r\n $scope.attributeUpdateFormGroup.markAsPristine();\r\n self.onDataUpdated();\r\n };\r\n \r\n $scope.disableButton = function () {\r\n return $scope.attributeUpdateFormGroup.get('currentLat').value === $scope.originalLat && $scope.attributeUpdateFormGroup.get('currentLng').value === $scope.originalLng || $scope.currentLng === null || $scope.currentLat === null;\r\n };\r\n \r\n $scope.getCoordinate = function() {\r\n if (navigator.geolocation) {\r\n navigator.geolocation.getCurrentPosition(showPosition, function (){\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.blocked-location'), \r\n 'bottom', 'left', $scope.toastTargetId);\r\n }, {\r\n enableHighAccuracy: settings.enableHighAccuracy\r\n });\r\n } else {\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.no-support-geolocation'), 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n };\r\n \r\n function showPosition(position) {\r\n $scope.attributeUpdateFormGroup.setValue({\r\n currentLat: correctValue(position.coords.latitude),\r\n currentLng: correctValue(position.coords.longitude)\r\n });\r\n $scope.attributeUpdateFormGroup.markAsDirty();\r\n $scope.isFocused = true;\r\n }\r\n \r\n self.onResize();\r\n}\r\n\r\nself.onDataUpdated = function() {\r\n try {\r\n if ($scope.dataKeyDetected) {\r\n if (!$scope.isFocused) {\r\n for(let i = 0; i < self.typeParameters().maxDataKeys; i++){\r\n if(self.ctx.data[i].dataKey.name === self.ctx.settings.latKeyName && $scope.attributeUpdateFormGroup.get('currentLat').pristine){\r\n $scope.originalLat = self.ctx.data[i].data[0][1];\r\n $scope.attributeUpdateFormGroup.get('currentLat').patchValue(correctValue($scope.originalLat));\r\n } else if(self.ctx.data[i].dataKey.name === self.ctx.settings.lngKeyName && $scope.attributeUpdateFormGroup.get('currentLng').pristine){\r\n $scope.originalLng = self.ctx.data[i].data[0][1];\r\n $scope.attributeUpdateFormGroup.get('currentLng').patchValue(correctValue($scope.originalLng));\r\n }\r\n }\r\n self.ctx.detectChanges();\r\n }\r\n }\r\n } catch (e) {\r\n console.log(e);\r\n }\r\n};\r\n\r\nfunction correctValue(value) {\r\n if (typeof value !== \"number\") {\r\n return 0;\r\n }\r\n return value;\r\n}\r\n\r\nself.onResize = function() {\r\n $scope.smallWidthContainer = (self.ctx.$container && self.ctx.$container[0].offsetWidth < 320);\r\n $scope.changeAlignment = ($scope.isHorizontal && self.ctx.$container && self.ctx.$container[0].offsetWidth < 480);\r\n self.ctx.detectChanges();\r\n};\r\n\r\nself.typeParameters = function() {\r\n return {\r\n maxDatasources: 1,\r\n maxDataKeys: 2,\r\n singleEntity: true\r\n };\r\n};\r\n\r\nself.onDestroy = function() {\r\n\r\n};",
  99 + "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"widgetTitle\": {\n \"title\": \"Widget title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"latKeyName\": {\n \"title\": \"Latitude key name\",\n \"type\": \"string\",\n \"default\": \"latitude\"\n },\n \"lngKeyName\": {\n \"title\": \"Longitude key name\",\n \"type\": \"string\",\n \"default\": \"longitude\"\n },\n \"showLabel\": {\n \"title\": \"Show label\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"latLabel\": {\n \"title\": \"Label for latitude\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"lngLabel\": {\n \"title\": \"Label for longitude\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"requiredErrorMessage\": {\n \"title\": \"'Required' error message\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"showResultMessage\": {\n \"title\": \"Show result message\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableHighAccuracy\": {\n \"title\": \"Use high accuracy\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"showGetLocation\": {\n \"title\": \"Show button 'Get current location'\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"inputFieldsAlignment\": {\n \"title\": \"Input fields alignment\",\n \"type\": \"string\",\n \"default\": \"column\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"widgetTitle\",\n \"latKeyName\",\n \"lngKeyName\",\n \"enableHighAccuracy\",\n \"showGetLocation\",\n \"showResultMessage\",\n {\n \"key\": \"inputFieldsAlignment\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"column\",\n \"label\": \"Column (default)\"\n },\n {\n \"value\": \"row\",\n \"label\": \"Row\"\n }\n ]\n },\n \"showLabel\",\n \"latLabel\",\n \"lngLabel\",\n \"requiredErrorMessage\"\n ]\n}",
20 100 "dataKeySettingsSchema": "{}\n",
21   - "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.05427416942713381,\"funcBody\":\"var value = prevValue || 15.833293;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.680594833308841,\"funcBody\":\"var value = prevValue || -90.454350;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]},{\"type\":\"function\",\"name\":\"Second point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#607d8b\",\"settings\":{},\"_hash\":0.7867521952070078,\"funcBody\":\"var value = prevValue || 14.450463;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#9c27b0\",\"settings\":{},\"_hash\":0.7040053227577256,\"funcBody\":\"var value = prevValue || -84.845334;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"fitMapBounds\":true,\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"label\":\"${entityName}\",\"tooltipPattern\":\"<b>${entityName}</b><br/><br/><b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><br/><link-act name='delete'>Delete</link-act>\",\"markerImageSize\":34,\"useColorFunction\":false,\"markerImages\":[],\"useMarkerImageFunction\":false,\"color\":\"#fe7569\",\"mapProvider\":\"OpenStreetMap.Mapnik\",\"showTooltip\":true,\"autocloseTooltip\":true,\"defaultCenterPosition\":\"0,0\",\"customProviderTileUrl\":\"https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png\",\"showTooltipAction\":\"click\",\"polygonKeyName\":\"coordinates\",\"polygonOpacity\":0.5,\"polygonStrokeOpacity\":1,\"polygonStrokeWeight\":1,\"zoomOnClick\":true,\"showCoverageOnHover\":true,\"animate\":true,\"maxClusterRadius\":80,\"removeOutsideVisibleBounds\":true,\"defaultZoomLevel\":5,\"provider\":\"openstreet-map\",\"draggableMarker\":true},\"title\":\"Markers Placement - OpenStreetMap\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{\"tooltipAction\":[{\"id\":\"54c293c4-9ca6-e34f-dc6a-0271944c1c66\",\"name\":\"delete\",\"icon\":\"more_horiz\",\"type\":\"custom\",\"customFunction\":\"var $rootScope = widgetContext.$scope.$injector.get('$rootScope');\\nvar entityDatasource = widgetContext.map.subscription.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id\\n });\\n\\nwidgetContext.map.saveMarkerLocation(entityDatasource[0],\\n widgetContext.map.locations[0], {\\n \\\"lat\\\": null,\\n \\\"lng\\\": null\\n }).then(function succes() {\\n $rootScope.$broadcast('widgetForceReInit');\\n });\"}]},\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"displayTimewindow\":true}"
  101 + "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Update shared location attribute\",\"dropShadow\":true,\"enableFullscreen\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
22 102 }
23 103 },
24 104 {
... ... @@ -54,22 +134,6 @@
54 134 }
55 135 },
56 136 {
57   - "alias": "markers_placement_image_map",
58   - "name": "Markers Placement - Image Map",
59   - "descriptor": {
60   - "type": "latest",
61   - "sizeX": 8.5,
62   - "sizeY": 6.5,
63   - "resources": [],
64   - "templateHtml": "",
65   - "templateCss": ".leaflet-zoom-box {\n\tz-index: 9;\n}\n\n.leaflet-pane { z-index: 4; }\n\n.leaflet-tile-pane { z-index: 2; }\n.leaflet-overlay-pane { z-index: 4; }\n.leaflet-shadow-pane { z-index: 5; }\n.leaflet-marker-pane { z-index: 6; }\n.leaflet-tooltip-pane { z-index: 7; }\n.leaflet-popup-pane { z-index: 8; }\n\n.leaflet-map-pane canvas { z-index: 1; }\n.leaflet-map-pane svg { z-index: 2; }\n\n.leaflet-control {\n\tz-index: 9;\n}\n.leaflet-top,\n.leaflet-bottom {\n\tz-index: 11;\n}\n\n.tb-marker-label {\n border: none;\n background: none;\n box-shadow: none;\n}\n\n.tb-marker-label:before {\n border: none;\n background: none;\n}\n",
66   - "controllerScript": "self.onInit = function() {\n self.ctx.map = new TbMapWidgetV2('image-map', false, self.ctx, null, true);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbMapWidgetV2.settingsSchema('image-map');\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbMapWidgetV2.dataKeySettingsSchema('image-map');\n}\n\nself.actionSources = function() {\n return TbMapWidgetV2.actionSources();\n}\n\nself.onDestroy = function() {\n}\n\nself.typeParameters = function() {\n return {\n hasDataPageLink: true\n };\n}",
67   - "settingsSchema": "{}",
68   - "dataKeySettingsSchema": "{}\n",
69   - "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"xPos\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.05427416942713381,\"funcBody\":\"var value = prevValue || 0.2;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"yPos\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.680594833308841,\"funcBody\":\"var value = prevValue || 0.3;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]},{\"type\":\"function\",\"name\":\"Second point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"xPos\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.05012157428742059,\"funcBody\":\"var value = prevValue || 0.6;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"yPos\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.6742359401617628,\"funcBody\":\"var value = prevValue || 0.7;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"showLabel\":true,\"label\":\"${entityName}\",\"tooltipPattern\":\"<b>${entityName}</b><br/><br/><b>X Pos:</b> ${xPos:2}<br/><b>Y Pos:</b> ${yPos:2}<br/><br/><link-act name='delete'>Delete</link-act>\",\"markerImageSize\":34,\"useColorFunction\":false,\"markerImages\":[],\"useMarkerImageFunction\":false,\"color\":\"#fe7569\",\"mapImageUrl\":\"\",\"xPosKeyName\":\"xPos\",\"yPosKeyName\":\"yPos\",\"posFunction\":\"return {x: origXPos, y: origYPos};\",\"markerOffsetX\":0.5,\"markerOffsetY\":1,\"showTooltip\":true,\"autocloseTooltip\":true,\"showTooltipAction\":\"click\",\"defaultCenterPosition\":\"0,0\",\"provider\":\"image-map\",\"fitMapBounds\":true,\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"polygonKeyName\":\"coordinates\",\"polygonOpacity\":0.5,\"polygonStrokeOpacity\":1,\"polygonStrokeWeight\":1,\"mapProvider\":\"HERE.normalDay\",\"draggableMarker\":true},\"title\":\"Markers Placement - Image Map\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{\"tooltipAction\":[{\"id\":\"c39f512a-21c6-6b06-3aa1-715262c6553d\",\"name\":\"delete\",\"icon\":\"more_horiz\",\"type\":\"custom\",\"customFunction\":\"var $rootScope = widgetContext.$scope.$injector.get('$rootScope');\\nvar entityDatasource = widgetContext.map.subscription.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id\\n });\\n\\nwidgetContext.map.saveMarkerLocation(entityDatasource[0],\\n widgetContext.map.locations[0], {\\n \\\"lat\\\": null,\\n \\\"lng\\\": null\\n }).then(function succes() {\\n $rootScope.$broadcast('widgetForceReInit');\\n });\"}]},\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"displayTimewindow\":true}"
70   - }
71   - },
72   - {
73 137 "alias": "update_integer_timeseries",
74 138 "name": "Update integer timeseries",
75 139 "descriptor": {
... ... @@ -278,83 +342,35 @@
278 342 }
279 343 },
280 344 {
281   - "alias": "update_location_timeseries",
282   - "name": "Update location timeseries",
283   - "descriptor": {
284   - "type": "latest",
285   - "sizeX": 7.5,
286   - "sizeY": 3,
287   - "resources": [],
288   - "templateHtml": "<div tb-toast toastTarget=\"{{ toastTargetId }}\" style=\"width: 100%; height: 100%;\">\n <form *ngIf=\"attributeUpdateFormGroup\"\n class=\"attribute-update-form\"\n [formGroup]=\"attributeUpdateFormGroup\"\n [ngClass]=\"{'small-width': smallWidthContainer}\"\n (ngSubmit)=\"updateAttribute()\">\n <div style=\"padding: 0 8px; margin: auto 0;\">\n <div class=\"attribute-update-form__grid\" [fxShow]=\"entityDetected && isValidParameter && dataKeyDetected\">\n <div class=\"grid__element\" \n [ngClass]=\"{'horizontal-alignment': isHorizontal && !changeAlignment}\">\n <mat-form-field class=\"mat-block\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? latLabel : '' }}</mat-label>\n <input matInput\n formControlName=\"currentLat\"\n required\n type=\"number\"\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n max=\"90\"\n min=\"-90\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentLat').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field>\n \n <mat-form-field class=\"mat-block\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? lngLabel : '' }}</mat-label>\n <input matInput\n formControlName=\"currentLng\"\n required\n type=\"number\"\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n max=\"180\"\n min=\"-180\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentLng').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field> \n </div>\n \n <div class=\"grid__element\">\n <button mat-button mat-icon-button class=\"getLocation\"\n type=\"button\"\n (click)=\"getCoordinate()\"\n *ngIf=\"settings.showGetLocation\"\n matTooltip=\"{{ 'widgets.input-widgets.get-location' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>my_location</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"applyChanges\"\n type=\"submit\"\n [disabled]=\"disableButton() || attributeUpdateFormGroup.invalid || attributeUpdateFormGroup.pristine\"\n matTooltip=\"{{ 'widgets.input-widgets.update-timeseries' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>check</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"discardChanges\"\n type=\"button\"\n [disabled]=\"disableButton()\"\n (click)=\"discardChange()\"\n matTooltip=\"{{ 'widgets.input-widgets.discard-changes' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>close</mat-icon>\n </button>\n </div>\n </div>\n \n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\" [fxHide]=\"entityDetected\" >\n {{ 'widgets.input-widgets.no-entity-selected' | translate }}\n </div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !dataKeyDetected\">\n {{ 'widgets.input-widgets.no-timeseries-selected' | translate }}\n </div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !isValidParameter\">\n {{ 'widgets.input-widgets.no-coordinate-specified' | translate }}\n </div>\n </div>\n </form>\n</div>",
289   - "templateCss": ".attribute-update-form {\n overflow: hidden;\n height: 100%;\n display: flex;\n flex-direction: column;\n}\n\n.attribute-update-form__grid {\n display: flex;\n}\n.grid__element:first-child {\n flex-direction: column;\n flex: 1;\n}\n\n.grid__element.horizontal-alignment {\n flex-direction: row;\n}\n\n.grid__element:last-child {\n align-items: center;\n margin-left: 7px;\n}\n.grid__element {\n display: flex;\n}\n\n.attribute-update-form .mat-button.mat-icon-button {\n margin: 0;\n}\n\n.attribute-update-form .mat-button.mat-icon-button {\n width: 32px;\n min-width: 32px;\n height: 32px;\n min-height: 32px;\n padding: 0 !important;\n margin: 0;\n line-height: 20px;\n}\n\n.attribute-update-form .mat-button.getLocation {\n margin-right: 10px;\n}\n\n.attribute-update-form .mat-icon-button mat-icon {\n width: 20px;\n min-width: 20px;\n height: 20px;\n min-height: 20px;\n font-size: 20px;\n}\n\n.attribute-update-form mat-form-field{\n width: 100%;\n padding-right: 5px;\n}\n\n.attribute-update-form.small-width mat-form-field{\n width: 150px;\n}\n\n.tb-toast {\n font-size: 14px!important;\n}",
290   - "controllerScript": "let $scope;\r\nlet settings;\r\nlet attributeService;\r\nlet utils;\r\nlet translate;\r\n\r\nself.onInit = function() {\r\n self.ctx.ngZone.run(function() {\r\n init(); \r\n self.ctx.detectChanges(true);\r\n });\r\n};\r\n\r\n\r\nfunction init() {\r\n $scope = self.ctx.$scope;\r\n attributeService = $scope.$injector.get(self.ctx.servicesMap.get('attributeService'));\r\n utils = $scope.$injector.get(self.ctx.servicesMap.get('utils'));\r\n translate = $scope.$injector.get(self.ctx.servicesMap.get('translate'));\r\n $scope.toastTargetId = 'input-widget' + utils.guid();\r\n settings = utils.deepClone(self.ctx.settings) || {};\r\n \r\n settings.showLabel = utils.defaultValue(settings.showLabel, true);\r\n settings.showResultMessage = utils.defaultValue(settings.showResultMessage, true);\r\n settings.showGetLocation = utils.defaultValue(settings.showGetLocation, true);\r\n settings.enableHighAccuracy = utils.defaultValue(settings.enableHighAccuracy, false);\r\n $scope.settings = settings;\r\n $scope.isValidParameter = true;\r\n $scope.dataKeyDetected = false; \r\n\r\n $scope.isHorizontal = (settings.inputFieldsAlignment === 'row');\r\n $scope.requiredErrorMessage = utils.customTranslation(settings.requiredErrorMessage, settings.requiredErrorMessage) || translate.instant('widgets.input-widgets.entity-coordinate-required');\r\n $scope.latLabel = utils.customTranslation(settings.latLabel, settings.latLabel) || translate.instant('widgets.input-widgets.latitude');\r\n $scope.lngLabel = utils.customTranslation(settings.lngLabel, settings.lngLabel) || translate.instant('widgets.input-widgets.longitude');\r\n\r\n $scope.attributeUpdateFormGroup = $scope.fb.group(\r\n {currentLat: [undefined, [$scope.validators.required,\r\n $scope.validators.min(-90),\r\n $scope.validators.max(90)]],\r\n currentLng: [undefined, [$scope.validators.required,\r\n $scope.validators.min(-180),\r\n $scope.validators.max(180)]]}\r\n );\r\n\r\n if (self.ctx.datasources && self.ctx.datasources.length) {\r\n var datasource = self.ctx.datasources[0];\r\n if (datasource.type === 'entity') {\r\n if (datasource.entityType && datasource.entityId) {\r\n $scope.entityName = datasource.entityName;\r\n if (settings.widgetTitle && settings.widgetTitle.length) {\r\n $scope.titleTemplate = utils.customTranslation(settings.widgetTitle, settings.widgetTitle);\r\n } else {\r\n $scope.titleTemplate = self.ctx.widgetConfig.title;\r\n }\r\n\r\n $scope.entityDetected = true;\r\n }\r\n }\r\n if (datasource.dataKeys.length > 1) {\r\n $scope.dataKeyDetected = true;\r\n for (let i = 0; i < datasource.dataKeys.length; i++) {\r\n if (datasource.dataKeys[i].type != \"timeseries\"){\r\n $scope.isValidParameter = false;\r\n }\r\n if (datasource.dataKeys[i].name !== settings.latKeyName && datasource.dataKeys[i].name !== settings.lngKeyName){\r\n $scope.dataKeyDetected = false;\r\n }\r\n }\r\n }\r\n }\r\n\r\n self.ctx.widgetTitle = utils.createLabelFromDatasource(self.ctx.datasources[0], $scope.titleTemplate);\r\n\r\n $scope.updateAttribute = function () {\r\n $scope.isFocused = false;\r\n if ($scope.entityDetected) {\r\n var datasource = self.ctx.datasources[0];\r\n\r\n attributeService.saveEntityTimeseries(\r\n datasource.entity.id,\r\n 'scope',\r\n [\r\n {\r\n key: settings.latKeyName,\r\n value: $scope.attributeUpdateFormGroup.get('currentLat').value\r\n },{\r\n key: settings.lngKeyName,\r\n value: $scope.attributeUpdateFormGroup.get('currentLng').value\r\n }\r\n ]\r\n ).subscribe(\r\n function success() {\r\n $scope.originalLat = $scope.attributeUpdateFormGroup.get('currentLat').value;\r\n $scope.originalLng = $scope.attributeUpdateFormGroup.get('currentLng').value;\r\n if (settings.showResultMessage) {\r\n $scope.showSuccessToast(translate.instant('widgets.input-widgets.update-successful'), 1000, 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n },\r\n function fail() {\r\n if (settings.showResultMessage) {\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.update-failed'), 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n }\r\n );\r\n }\r\n };\r\n\r\n $scope.changeFocus = function () {\r\n if ($scope.attributeUpdateFormGroup.get('currentLat').value === $scope.originalLat && $scope.attributeUpdateFormGroup.get('currentLng').value === $scope.originalLng) {\r\n $scope.isFocused = false;\r\n }\r\n };\r\n \r\n $scope.discardChange = function() {\r\n $scope.attributeUpdateFormGroup.setValue({\r\n 'currentLat': $scope.originalLat,\r\n 'currentLng': $scope.originalLng\r\n });\r\n $scope.isFocused = false;\r\n $scope.attributeUpdateFormGroup.markAsPristine();\r\n self.onDataUpdated();\r\n };\r\n \r\n $scope.disableButton = function () {\r\n return $scope.attributeUpdateFormGroup.get('currentLat').value === $scope.originalLat && $scope.attributeUpdateFormGroup.get('currentLng').value === $scope.originalLng || $scope.currentLng === null || $scope.currentLat === null;\r\n };\r\n \r\n $scope.getCoordinate = function() {\r\n if (navigator.geolocation) {\r\n navigator.geolocation.getCurrentPosition(showPosition, function (){\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.blocked-location'), \r\n 'bottom', 'left', $scope.toastTargetId);\r\n }, {\r\n enableHighAccuracy: settings.enableHighAccuracy\r\n });\r\n } else {\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.no-support-geolocation'), 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n };\r\n \r\n function showPosition(position) {\r\n $scope.attributeUpdateFormGroup.setValue({\r\n currentLat: correctValue(position.coords.latitude),\r\n currentLng: correctValue(position.coords.longitude)\r\n });\r\n $scope.attributeUpdateFormGroup.markAsDirty();\r\n $scope.isFocused = true;\r\n }\r\n \r\n self.onResize();\r\n}\r\n\r\nself.onDataUpdated = function() {\r\n try {\r\n if ($scope.dataKeyDetected) {\r\n if (!$scope.isFocused) {\r\n for(let i = 0; i < self.typeParameters().maxDataKeys; i++){\r\n if(self.ctx.data[i].dataKey.name === self.ctx.settings.latKeyName && $scope.attributeUpdateFormGroup.get('currentLat').pristine){\r\n $scope.originalLat = self.ctx.data[i].data[0][1];\r\n $scope.attributeUpdateFormGroup.get('currentLat').patchValue(correctValue($scope.originalLat));\r\n } else if(self.ctx.data[i].dataKey.name === self.ctx.settings.lngKeyName && $scope.attributeUpdateFormGroup.get('currentLng').pristine){\r\n $scope.originalLng = self.ctx.data[i].data[0][1];\r\n $scope.attributeUpdateFormGroup.get('currentLng').patchValue(correctValue($scope.originalLng));\r\n }\r\n }\r\n self.ctx.detectChanges();\r\n }\r\n }\r\n } catch (e) {\r\n console.log(e);\r\n }\r\n};\r\n\r\nfunction correctValue(value) {\r\n if (typeof value !== \"number\") {\r\n return 0;\r\n }\r\n return value;\r\n}\r\n\r\nself.onResize = function() {\r\n $scope.smallWidthContainer = (self.ctx.$container && self.ctx.$container[0].offsetWidth < 320);\r\n $scope.changeAlignment = ($scope.isHorizontal && self.ctx.$container && self.ctx.$container[0].offsetWidth < 480);\r\n self.ctx.detectChanges();\r\n};\r\n\r\nself.typeParameters = function() {\r\n return {\r\n maxDatasources: 1,\r\n maxDataKeys: 2,\r\n singleEntity: true\r\n };\r\n};\r\n\r\nself.onDestroy = function() {\r\n\r\n};",
291   - "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"widgetTitle\": {\n \"title\": \"Widget title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"latKeyName\": {\n \"title\": \"Latitude key name\",\n \"type\": \"string\",\n \"default\": \"latitude\"\n },\n \"lngKeyName\": {\n \"title\": \"Longitude key name\",\n \"type\": \"string\",\n \"default\": \"longitude\"\n },\n \"showLabel\": {\n \"title\": \"Show label\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"latLabel\": {\n \"title\": \"Label for latitude\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"lngLabel\": {\n \"title\": \"Label for longitude\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"requiredErrorMessage\": {\n \"title\": \"'Required' error message\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"showResultMessage\": {\n \"title\": \"Show result message\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableHighAccuracy\": {\n \"title\": \"Use high accuracy\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"showGetLocation\": {\n \"title\": \"Show button 'Get current location'\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"inputFieldsAlignment\": {\n \"title\": \"Input fields alignment\",\n \"type\": \"string\",\n \"default\": \"column\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"widgetTitle\",\n \"latKeyName\",\n \"lngKeyName\",\n \"enableHighAccuracy\",\n \"showGetLocation\",\n \"showResultMessage\",\n {\n \"key\": \"inputFieldsAlignment\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"column\",\n \"label\": \"Column (default)\"\n },\n {\n \"value\": \"row\",\n \"label\": \"Row\"\n }\n ]\n },\n \"showLabel\",\n \"latLabel\",\n \"lngLabel\",\n \"requiredErrorMessage\"\n ]\n}",
292   - "dataKeySettingsSchema": "{}\n",
293   - "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Update location timeseries\",\"dropShadow\":true,\"enableFullscreen\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
294   - }
295   - },
296   - {
297   - "alias": "update_shared_double_attribute",
298   - "name": "Update shared double attribute",
299   - "descriptor": {
300   - "type": "latest",
301   - "sizeX": 7.5,
302   - "sizeY": 3,
303   - "resources": [],
304   - "templateHtml": "<div tb-toast toastTarget=\"{{ toastTargetId }}\" style=\"width: 100%; height: 100%;\">\n <form *ngIf=\"attributeUpdateFormGroup\"\n class=\"attribute-update-form\"\n [formGroup]=\"attributeUpdateFormGroup\"\n (ngSubmit)=\"updateAttribute()\">\n <div style=\"padding: 0 8px; margin: auto 0;\">\n <div class=\"attribute-update-form__grid\" [fxShow]=\"entityDetected && isValidParameter && dataKeyDetected\">\n <div class=\"grid__element\">\n <mat-form-field class=\"mat-block\" style=\"width: 100%;\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? labelValue : '' }}</mat-label>\n <input matInput\n formControlName=\"currentValue\"\n required\n type=\"number\"\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n max=\"{{settings.maxValue}}\"\n min=\"{{settings.minValue}}\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentValue').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field> \n </div>\n \n <div class=\"grid__element\">\n <button mat-button mat-icon-button class=\"applyChanges\"\n type=\"submit\"\n [disabled]=\"originalValue === attributeUpdateFormGroup.get('currentValue').value || attributeUpdateFormGroup.invalid || !attributeUpdateFormGroup.dirty\"\n matTooltip=\"{{ 'widgets.input-widgets.update-attribute' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>check</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"discardChanges\"\n type=\"button\"\n [disabled]=\"originalValue === attributeUpdateFormGroup.get('currentValue').value\"\n (click)=\"attributeUpdateFormGroup.get('currentValue').patchValue(originalValue); isFocused = false\"\n matTooltip=\"{{ 'widgets.input-widgets.discard-changes' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>close</mat-icon>\n </button>\n </div>\n </div>\n \n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\" [fxHide]=\"entityDetected\" [innerHtml]=\"message\"></div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !dataKeyDetected\">\n {{ 'widgets.input-widgets.no-attribute-selected' | translate }}\n </div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !isValidParameter\">\n {{ 'widgets.input-widgets.timeseries-not-allowed' | translate }}\n </div>\n </div>\n </form>\n</div>",
305   - "templateCss": ".attribute-update-form {\n overflow: hidden;\n height: 100%;\n display: flex;\n flex-direction: column;\n}\n\n.attribute-update-form__grid {\n display: flex;\n}\n.grid__element:first-child {\n flex: 1;\n}\n.grid__element:last-child {\n margin-top: 19px;\n margin-left: 7px;\n}\n.grid__element {\n display: flex;\n}\n\n.attribute-update-form .mat-button.mat-icon-button {\n width: 32px;\n min-width: 32px;\n height: 32px;\n min-height: 32px;\n padding: 0 !important;\n margin: 0 !important;\n line-height: 20px;\n}\n\n.attribute-update-form .mat-icon-button mat-icon {\n width: 20px;\n min-width: 20px;\n height: 20px;\n min-height: 20px;\n font-size: 20px;\n}\n\n.tb-toast {\n font-size: 14px!important;\n}",
306   - "controllerScript": "let $scope;\nlet settings;\nlet attributeService;\nlet utils;\nlet translate;\n\nself.onInit = function() {\n self.ctx.ngZone.run(function() {\n init(); \n self.ctx.detectChanges(true);\n });\n};\n\n\nfunction init() {\n\n $scope = self.ctx.$scope;\n attributeService = $scope.$injector.get(self.ctx.servicesMap.get('attributeService'));\n utils = $scope.$injector.get(self.ctx.servicesMap.get('utils'));\n translate = $scope.$injector.get(self.ctx.servicesMap.get('translate'));\n $scope.toastTargetId = 'input-widget' + utils.guid();\n settings = utils.deepClone(self.ctx.settings) || {};\n settings.showLabel = utils.defaultValue(settings.showLabel, true);\n settings.showResultMessage = utils.defaultValue(settings.showResultMessage, true);\n $scope.settings = settings;\n $scope.isValidParameter = true;\n $scope.dataKeyDetected = false; \n $scope.message = translate.instant('widgets.input-widgets.no-entity-selected');\n \n $scope.requiredErrorMessage = utils.customTranslation(settings.requiredErrorMessage, settings.requiredErrorMessage) || translate.instant('widgets.input-widgets.entity-attribute-required');\n $scope.labelValue = utils.customTranslation(settings.labelValue, settings.labelValue) || translate.instant('widgets.input-widgets.value');\n\n $scope.attributeUpdateFormGroup = $scope.fb.group(\n {currentValue: [undefined, [$scope.validators.required,\n $scope.validators.min(settings.minValue),\n $scope.validators.max(settings.maxValue)]]}\n );\n\n if (self.ctx.datasources && self.ctx.datasources.length) {\n var datasource = self.ctx.datasources[0];\n if (datasource.type === 'entity') {\n if (datasource.entityType === 'DEVICE') {\n if (datasource.entityType && datasource.entityId) {\n $scope.entityName = datasource.entityName;\n if (settings.widgetTitle && settings.widgetTitle.length) {\n $scope.titleTemplate = utils.customTranslation(settings.widgetTitle, settings.widgetTitle);\n } else {\n $scope.titleTemplate = self.ctx.widgetConfig.title;\n }\n \n $scope.entityDetected = true;\n }\n } else {\n $scope.message = translate.instant('widgets.input-widgets.not-allowed-entity');\n }\n }\n if (datasource.dataKeys.length) {\n if (datasource.dataKeys[0].type !== \"attribute\") {\n $scope.isValidParameter = false;\n } else {\n $scope.currentKey = datasource.dataKeys[0].name;\n $scope.dataKeyType = datasource.dataKeys[0].type;\n $scope.dataKeyDetected = true;\n }\n }\n }\n\n self.ctx.widgetTitle = utils.createLabelFromDatasource(self.ctx.datasources[0], $scope.titleTemplate);\n\n $scope.updateAttribute = function () {\n $scope.isFocused = false;\n if ($scope.entityDetected) {\n var datasource = self.ctx.datasources[0];\n\n attributeService.saveEntityAttributes(\n datasource.entity.id,\n 'SHARED_SCOPE',\n [\n {\n key: $scope.currentKey,\n value: $scope.attributeUpdateFormGroup.get('currentValue').value\n }\n ]\n ).subscribe(\n function success() {\n $scope.originalValue = $scope.attributeUpdateFormGroup.get('currentValue').value;\n if (settings.showResultMessage) {\n $scope.showSuccessToast(translate.instant('widgets.input-widgets.update-successful'), 1000, 'bottom', 'left', $scope.toastTargetId);\n }\n },\n function fail() {\n if (settings.showResultMessage) {\n $scope.showErrorToast(translate.instant('widgets.input-widgets.update-failed'), 'bottom', 'left', $scope.toastTargetId);\n }\n }\n );\n }\n };\n\n $scope.changeFocus = function () {\n if ($scope.attributeUpdateFormGroup.get('currentValue').value === $scope.originalValue) {\n $scope.isFocused = false;\n }\n }\n}\n\nself.onDataUpdated = function() {\n\n try {\n if ($scope.dataKeyDetected) {\n if (!$scope.isFocused) {\n $scope.originalValue = self.ctx.data[0].data[0][1];\n $scope.attributeUpdateFormGroup.get('currentValue').patchValue(correctValue($scope.originalValue));\n self.ctx.detectChanges();\n }\n }\n } catch (e) {\n console.log(e);\n }\n}\n\nfunction correctValue(value) {\n if (typeof value !== \"number\") {\n return 0;\n }\n return value;\n}\n\nself.onResize = function() {\n\n}\n\nself.typeParameters = function() {\n return {\n maxDatasources: 1,\n maxDataKeys: 1,\n singleEntity: true\n }\n}\n\nself.onDestroy = function() {\n\n}",
307   - "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"widgetTitle\": {\n \"title\": \"Widget title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"showLabel\":{\n \"title\":\"Show label\",\n \"type\":\"boolean\",\n \"default\":true\n },\n \"labelValue\": {\n \"title\": \"Label\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"requiredErrorMessage\": {\n \"title\": \"'Required' error message\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"maxValue\": {\n \"title\": \"Max value\",\n \"type\": \"number\",\n \"default\": \"\"\n },\n \"minValue\": {\n \"title\": \"Min value\",\n \"type\": \"number\",\n \"default\": \"\"\n },\n \"showResultMessage\":{\n \"title\":\"Show result message\",\n \"type\":\"boolean\",\n \"default\":true\n }\n },\n \"required\": []\n },\n \"form\": [\n \"widgetTitle\",\n \"showResultMessage\",\n \"showLabel\",\n \"labelValue\",\n \"requiredErrorMessage\",\n \"maxValue\",\n \"minValue\"\n ]\n}",
308   - "dataKeySettingsSchema": "{}\n",
309   - "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Update shared double attribute\",\"dropShadow\":true,\"enableFullscreen\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
310   - }
311   - },
312   - {
313   - "alias": "update_shared_integer_attribute",
314   - "name": "Update shared integer attribute",
315   - "descriptor": {
316   - "type": "latest",
317   - "sizeX": 7.5,
318   - "sizeY": 3,
319   - "resources": [],
320   - "templateHtml": "<div tb-toast toastTarget=\"{{ toastTargetId }}\" style=\"width: 100%; height: 100%;\">\n <form *ngIf=\"attributeUpdateFormGroup\"\n class=\"attribute-update-form\"\n [formGroup]=\"attributeUpdateFormGroup\"\n (ngSubmit)=\"updateAttribute()\">\n <div style=\"padding: 0 8px; margin: auto 0;\">\n <div class=\"attribute-update-form__grid\" [fxShow]=\"entityDetected && isValidParameter && dataKeyDetected\">\n <div class=\"grid__element\">\n <mat-form-field class=\"mat-block\" style=\"width: 100%;\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? labelValue : '' }}</mat-label>\n <input matInput\n formControlName=\"currentValue\"\n required\n type=\"number\"\n step=\"1\"\n pattern=\"^-?[0-9]+$\"\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n max=\"{{settings.maxValue}}\"\n min=\"{{settings.minValue}}\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentValue').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field> \n </div>\n \n <div class=\"grid__element\">\n <button mat-button mat-icon-button class=\"applyChanges\"\n type=\"submit\"\n [disabled]=\"originalValue === attributeUpdateFormGroup.get('currentValue').value || attributeUpdateFormGroup.invalid || !attributeUpdateFormGroup.dirty\"\n matTooltip=\"{{ 'widgets.input-widgets.update-attribute' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>check</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"discardChanges\"\n type=\"button\"\n [disabled]=\"originalValue === attributeUpdateFormGroup.get('currentValue').value\"\n (click)=\"attributeUpdateFormGroup.get('currentValue').patchValue(originalValue); isFocused = false\"\n matTooltip=\"{{ 'widgets.input-widgets.discard-changes' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>close</mat-icon>\n </button>\n </div>\n </div>\n \n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\" [fxHide]=\"entityDetected\" [innerHtml]=\"message\"></div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !dataKeyDetected\">\n {{ 'widgets.input-widgets.no-attribute-selected' | translate }}\n </div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !isValidParameter\">\n {{ 'widgets.input-widgets.timeseries-not-allowed' | translate }}\n </div>\n </div>\n </form>\n</div>",
321   - "templateCss": ".attribute-update-form {\n overflow: hidden;\n height: 100%;\n display: flex;\n flex-direction: column;\n}\n\n.attribute-update-form__grid {\n display: flex;\n}\n.grid__element:first-child {\n flex: 1;\n}\n.grid__element:last-child {\n margin-top: 19px;\n margin-left: 7px;\n}\n.grid__element {\n display: flex;\n}\n\n.attribute-update-form .mat-button.mat-icon-button {\n width: 32px;\n min-width: 32px;\n height: 32px;\n min-height: 32px;\n padding: 0 !important;\n margin: 0 !important;\n line-height: 20px;\n}\n\n.attribute-update-form .mat-icon-button mat-icon {\n width: 20px;\n min-width: 20px;\n height: 20px;\n min-height: 20px;\n font-size: 20px;\n}\n\n.tb-toast {\n font-size: 14px!important;\n}",
322   - "controllerScript": "let $scope;\nlet settings;\nlet attributeService;\nlet utils;\nlet translate;\n\nself.onInit = function() {\n self.ctx.ngZone.run(function() {\n init(); \n self.ctx.detectChanges(true);\n });\n};\n\n\nfunction init() {\n\n $scope = self.ctx.$scope;\n attributeService = $scope.$injector.get(self.ctx.servicesMap.get('attributeService'));\n utils = $scope.$injector.get(self.ctx.servicesMap.get('utils'));\n translate = $scope.$injector.get(self.ctx.servicesMap.get('translate'));\n $scope.toastTargetId = 'input-widget' + utils.guid();\n settings = utils.deepClone(self.ctx.settings) || {};\n settings.showLabel = utils.defaultValue(settings.showLabel, true);\n settings.showResultMessage = utils.defaultValue(settings.showResultMessage, true);\n $scope.settings = settings;\n $scope.isValidParameter = true;\n $scope.dataKeyDetected = false; \n $scope.message = translate.instant('widgets.input-widgets.no-entity-selected');\n \n $scope.requiredErrorMessage = utils.customTranslation(settings.requiredErrorMessage, settings.requiredErrorMessage) || translate.instant('widgets.input-widgets.entity-attribute-required');\n $scope.labelValue = utils.customTranslation(settings.labelValue, settings.labelValue) || translate.instant('widgets.input-widgets.value');\n\n $scope.attributeUpdateFormGroup = $scope.fb.group(\n {currentValue: [undefined, [$scope.validators.required,\n $scope.validators.min(settings.minValue),\n $scope.validators.max(settings.maxValue),\n $scope.validators.pattern(/^-?[0-9]+$/)]]}\n );\n\n if (self.ctx.datasources && self.ctx.datasources.length) {\n var datasource = self.ctx.datasources[0];\n if (datasource.type === 'entity') {\n if (datasource.entityType === 'DEVICE') {\n if (datasource.entityType && datasource.entityId) {\n $scope.entityName = datasource.entityName;\n if (settings.widgetTitle && settings.widgetTitle.length) {\n $scope.titleTemplate = utils.customTranslation(settings.widgetTitle, settings.widgetTitle);\n } else {\n $scope.titleTemplate = self.ctx.widgetConfig.title;\n }\n \n $scope.entityDetected = true;\n }\n } else {\n $scope.message = translate.instant('widgets.input-widgets.not-allowed-entity');\n }\n }\n if (datasource.dataKeys.length) {\n if (datasource.dataKeys[0].type !== \"attribute\") {\n $scope.isValidParameter = false;\n } else {\n $scope.currentKey = datasource.dataKeys[0].name;\n $scope.dataKeyType = datasource.dataKeys[0].type;\n $scope.dataKeyDetected = true;\n }\n }\n }\n\n self.ctx.widgetTitle = utils.createLabelFromDatasource(self.ctx.datasources[0], $scope.titleTemplate);\n\n $scope.updateAttribute = function () {\n $scope.isFocused = false;\n if ($scope.entityDetected) {\n var datasource = self.ctx.datasources[0];\n\n attributeService.saveEntityAttributes(\n datasource.entity.id,\n 'SHARED_SCOPE',\n [\n {\n key: $scope.currentKey,\n value: $scope.attributeUpdateFormGroup.get('currentValue').value\n }\n ]\n ).subscribe(\n function success() {\n $scope.originalValue = $scope.attributeUpdateFormGroup.get('currentValue').value;\n if (settings.showResultMessage) {\n $scope.showSuccessToast(translate.instant('widgets.input-widgets.update-successful'), 1000, 'bottom', 'left', $scope.toastTargetId);\n }\n },\n function fail() {\n if (settings.showResultMessage) {\n $scope.showErrorToast(translate.instant('widgets.input-widgets.update-failed'), 'bottom', 'left', $scope.toastTargetId);\n }\n }\n );\n }\n };\n\n $scope.changeFocus = function () {\n if ($scope.attributeUpdateFormGroup.get('currentValue').value === $scope.originalValue) {\n $scope.isFocused = false;\n }\n }\n}\n\nself.onDataUpdated = function() {\n\n try {\n if ($scope.dataKeyDetected) {\n if (!$scope.isFocused) {\n $scope.originalValue = self.ctx.data[0].data[0][1];\n $scope.attributeUpdateFormGroup.get('currentValue').patchValue(correctValue($scope.originalValue));\n self.ctx.detectChanges();\n }\n }\n } catch (e) {\n console.log(e);\n }\n}\n\nfunction correctValue(value) {\n if (typeof value !== \"number\") {\n return 0;\n }\n return value;\n}\n\nself.onResize = function() {\n\n}\n\nself.typeParameters = function() {\n return {\n maxDatasources: 1,\n maxDataKeys: 1,\n singleEntity: true\n }\n}\n\nself.onDestroy = function() {\n\n}",
323   - "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"widgetTitle\": {\n \"title\": \"Widget title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"showLabel\":{\n \"title\":\"Show label\",\n \"type\":\"boolean\",\n \"default\":true\n },\n \"labelValue\": {\n \"title\": \"Label\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"requiredErrorMessage\": {\n \"title\": \"'Required' error message\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"maxValue\": {\n \"title\": \"Max value\",\n \"type\": \"number\",\n \"default\": \"\"\n },\n \"minValue\": {\n \"title\": \"Min value\",\n \"type\": \"number\",\n \"default\": \"\"\n },\n \"showResultMessage\":{\n \"title\":\"Show result message\",\n \"type\":\"boolean\",\n \"default\":true\n }\n },\n \"required\": []\n },\n \"form\": [\n \"widgetTitle\",\n \"showResultMessage\",\n \"showLabel\",\n \"labelValue\",\n \"requiredErrorMessage\",\n \"maxValue\",\n \"minValue\"\n ]\n}",
324   - "dataKeySettingsSchema": "{}\n",
325   - "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Update shared integer attribute\",\"dropShadow\":true,\"enableFullscreen\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
326   - }
327   - },
328   - {
329   - "alias": "update_shared_string_attribute",
330   - "name": "Update shared string attribute",
  345 + "alias": "markers_placement_image_map",
  346 + "name": "Markers Placement - Image Map",
331 347 "descriptor": {
332 348 "type": "latest",
333   - "sizeX": 7.5,
334   - "sizeY": 3,
  349 + "sizeX": 8.5,
  350 + "sizeY": 6.5,
335 351 "resources": [],
336   - "templateHtml": "<div tb-toast toastTarget=\"{{ toastTargetId }}\" style=\"width: 100%; height: 100%;\">\n <form *ngIf=\"attributeUpdateFormGroup\"\n class=\"attribute-update-form\"\n [formGroup]=\"attributeUpdateFormGroup\"\n (ngSubmit)=\"updateAttribute()\">\n <div style=\"padding: 0 8px; margin: auto 0;\">\n <div class=\"attribute-update-form__grid\" [fxShow]=\"entityDetected && isValidParameter && dataKeyDetected\">\n <div class=\"grid__element\">\n <mat-form-field class=\"mat-block\" style=\"width: 100%;\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? labelValue : '' }}</mat-label>\n <input matInput\n formControlName=\"currentValue\"\n required\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n maxlength=\"{{settings.maxLength}}\"\n minlength=\"{{settings.minLength}}\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentValue').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field> \n </div>\n \n <div class=\"grid__element\">\n <button mat-button mat-icon-button class=\"applyChanges\"\n type=\"submit\"\n [disabled]=\"originalValue === attributeUpdateFormGroup.get('currentValue').value || attributeUpdateFormGroup.invalid || !attributeUpdateFormGroup.dirty\"\n matTooltip=\"{{ 'widgets.input-widgets.update-attribute' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>check</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"discardChanges\"\n type=\"button\"\n [disabled]=\"originalValue === attributeUpdateFormGroup.get('currentValue').value\"\n (click)=\"attributeUpdateFormGroup.get('currentValue').patchValue(originalValue); isFocused = false\"\n matTooltip=\"{{ 'widgets.input-widgets.discard-changes' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>close</mat-icon>\n </button>\n </div>\n </div>\n \n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\" [fxHide]=\"entityDetected\" [innerHtml]=\"message\"></div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !dataKeyDetected\">\n {{ 'widgets.input-widgets.no-attribute-selected' | translate }}\n </div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !isValidParameter\">\n {{ 'widgets.input-widgets.timeseries-not-allowed' | translate }}\n </div>\n </div>\n </form>\n</div>",
337   - "templateCss": ".attribute-update-form {\n overflow: hidden;\n height: 100%;\n display: flex;\n flex-direction: column;\n}\n\n.attribute-update-form__grid {\n display: flex;\n}\n.grid__element:first-child {\n flex: 1;\n}\n.grid__element:last-child {\n margin-top: 19px;\n margin-left: 7px;\n}\n.grid__element {\n display: flex;\n}\n\n.attribute-update-form .mat-button.mat-icon-button {\n width: 32px;\n min-width: 32px;\n height: 32px;\n min-height: 32px;\n padding: 0 !important;\n margin: 0 !important;\n line-height: 20px;\n}\n\n.attribute-update-form .mat-icon-button mat-icon {\n width: 20px;\n min-width: 20px;\n height: 20px;\n min-height: 20px;\n font-size: 20px;\n}\n\n.tb-toast {\n font-size: 14px!important;\n}",
338   - "controllerScript": "let $scope;\nlet settings;\nlet attributeService;\nlet utils;\nlet translate;\n\nself.onInit = function() {\n self.ctx.ngZone.run(function() {\n init(); \n self.ctx.detectChanges(true);\n });\n};\n\n\nfunction init() {\n\n $scope = self.ctx.$scope;\n attributeService = $scope.$injector.get(self.ctx.servicesMap.get('attributeService'));\n utils = $scope.$injector.get(self.ctx.servicesMap.get('utils'));\n translate = $scope.$injector.get(self.ctx.servicesMap.get('translate'));\n $scope.toastTargetId = 'input-widget' + utils.guid();\n settings = utils.deepClone(self.ctx.settings) || {};\n settings.showLabel = utils.defaultValue(settings.showLabel, true);\n settings.showResultMessage = utils.defaultValue(settings.showResultMessage, true);\n $scope.settings = settings;\n $scope.isValidParameter = true;\n $scope.dataKeyDetected = false; \n $scope.message = translate.instant('widgets.input-widgets.no-entity-selected');\n \n $scope.requiredErrorMessage = utils.customTranslation(settings.requiredErrorMessage, settings.requiredErrorMessage) || translate.instant('widgets.input-widgets.entity-attribute-required');\n $scope.labelValue = utils.customTranslation(settings.labelValue, settings.labelValue) || translate.instant('widgets.input-widgets.value');\n\n $scope.attributeUpdateFormGroup = $scope.fb.group(\n {currentValue: [undefined, [$scope.validators.required,\n $scope.validators.minLength(settings.minLength),\n $scope.validators.maxLength(settings.maxLength)]]}\n );\n\n if (self.ctx.datasources && self.ctx.datasources.length) {\n var datasource = self.ctx.datasources[0];\n if (datasource.type === 'entity') {\n if (datasource.entityType === 'DEVICE') {\n if (datasource.entityType && datasource.entityId) {\n $scope.entityName = datasource.entityName;\n if (settings.widgetTitle && settings.widgetTitle.length) {\n $scope.titleTemplate = utils.customTranslation(settings.widgetTitle, settings.widgetTitle);\n } else {\n $scope.titleTemplate = self.ctx.widgetConfig.title;\n }\n \n $scope.entityDetected = true;\n }\n } else {\n $scope.message = translate.instant('widgets.input-widgets.not-allowed-entity');\n }\n }\n if (datasource.dataKeys.length) {\n if (datasource.dataKeys[0].type !== \"attribute\") {\n $scope.isValidParameter = false;\n } else {\n $scope.currentKey = datasource.dataKeys[0].name;\n $scope.dataKeyType = datasource.dataKeys[0].type;\n $scope.dataKeyDetected = true;\n }\n }\n }\n\n self.ctx.widgetTitle = utils.createLabelFromDatasource(self.ctx.datasources[0], $scope.titleTemplate);\n\n $scope.updateAttribute = function () {\n $scope.isFocused = false;\n if ($scope.entityDetected) {\n var datasource = self.ctx.datasources[0];\n\n attributeService.saveEntityAttributes(\n datasource.entity.id,\n 'SHARED_SCOPE',\n [\n {\n key: $scope.currentKey,\n value: $scope.attributeUpdateFormGroup.get('currentValue').value\n }\n ]\n ).subscribe(\n function success() {\n $scope.originalValue = $scope.attributeUpdateFormGroup.get('currentValue').value;\n if (settings.showResultMessage) {\n $scope.showSuccessToast(translate.instant('widgets.input-widgets.update-successful'), 1000, 'bottom', 'left', $scope.toastTargetId);\n }\n },\n function fail() {\n if (settings.showResultMessage) {\n $scope.showErrorToast(translate.instant('widgets.input-widgets.update-failed'), 'bottom', 'left', $scope.toastTargetId);\n }\n }\n );\n }\n };\n\n $scope.changeFocus = function () {\n if ($scope.attributeUpdateFormGroup.get('currentValue').value === $scope.originalValue) {\n $scope.isFocused = false;\n }\n }\n}\n\nself.onDataUpdated = function() {\n try {\n if ($scope.dataKeyDetected) {\n if (!$scope.isFocused) {\n $scope.originalValue = self.ctx.data[0].data[0][1];\n $scope.attributeUpdateFormGroup.get('currentValue').patchValue($scope.originalValue);\n self.ctx.detectChanges();\n }\n }\n } catch (e) {\n console.log(e);\n }\n}\n\nself.onResize = function() {\n\n}\n\nself.typeParameters = function() {\n return {\n maxDatasources: 1,\n maxDataKeys: 1,\n singleEntity: true\n }\n}\n\nself.onDestroy = function() {\n\n}",
339   - "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"widgetTitle\": {\n \"title\": \"Widget title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"showLabel\":{\n \"title\":\"Show label\",\n \"type\":\"boolean\",\n \"default\":true\n },\n \"labelValue\": {\n \"title\": \"Label\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"requiredErrorMessage\": {\n \"title\": \"'Required' error message\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"maxLength\": {\n \"title\": \"Max length\",\n \"type\": \"number\",\n \"default\": \"\"\n },\n \"minLength\": {\n \"title\": \"Min length\",\n \"type\": \"number\",\n \"default\": \"\"\n },\n \"showResultMessage\":{\n \"title\":\"Show result message\",\n \"type\":\"boolean\",\n \"default\":true\n }\n },\n \"required\": []\n },\n \"form\": [\n \"widgetTitle\",\n \"showResultMessage\",\n \"showLabel\",\n \"labelValue\",\n \"requiredErrorMessage\",\n \"maxLength\",\n \"minLength\"\n ]\n}",
  352 + "templateHtml": "",
  353 + "templateCss": ".leaflet-zoom-box {\n\tz-index: 9;\n}\n\n.leaflet-pane { z-index: 4; }\n\n.leaflet-tile-pane { z-index: 2; }\n.leaflet-overlay-pane { z-index: 4; }\n.leaflet-shadow-pane { z-index: 5; }\n.leaflet-marker-pane { z-index: 6; }\n.leaflet-tooltip-pane { z-index: 7; }\n.leaflet-popup-pane { z-index: 8; }\n\n.leaflet-map-pane canvas { z-index: 1; }\n.leaflet-map-pane svg { z-index: 2; }\n\n.leaflet-control {\n\tz-index: 9;\n}\n.leaflet-top,\n.leaflet-bottom {\n\tz-index: 11;\n}\n\n.tb-marker-label {\n border: none;\n background: none;\n box-shadow: none;\n}\n\n.tb-marker-label:before {\n border: none;\n background: none;\n}\n",
  354 + "controllerScript": "self.onInit = function() {\n self.ctx.map = new TbMapWidgetV2('image-map', false, self.ctx, null, true);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbMapWidgetV2.settingsSchema('image-map');\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbMapWidgetV2.dataKeySettingsSchema('image-map');\n}\n\nself.actionSources = function() {\n return TbMapWidgetV2.actionSources();\n}\n\nself.onDestroy = function() {\n}\n\nself.typeParameters = function() {\n return {\n hasDataPageLink: true\n };\n}",
  355 + "settingsSchema": "{}",
340 356 "dataKeySettingsSchema": "{}\n",
341   - "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Sin\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.23592248334107624,\"funcBody\":\"return Math.round(1000*Math.sin(time/5000));\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Update shared string attribute\",\"dropShadow\":true,\"enableFullscreen\":false,\"enableDataExport\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
  357 + "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"xPos\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.05427416942713381,\"funcBody\":\"var value = prevValue || 0.2;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"yPos\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.680594833308841,\"funcBody\":\"var value = prevValue || 0.3;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]},{\"type\":\"function\",\"name\":\"Second point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"xPos\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.05012157428742059,\"funcBody\":\"var value = prevValue || 0.6;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"yPos\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.6742359401617628,\"funcBody\":\"var value = prevValue || 0.7;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"showLabel\":true,\"label\":\"${entityName}\",\"tooltipPattern\":\"<b>${entityName}</b><br/><br/><b>X Pos:</b> ${xPos:2}<br/><b>Y Pos:</b> ${yPos:2}<br/><br/><link-act name='delete'>Delete</link-act>\",\"markerImageSize\":34,\"useColorFunction\":false,\"markerImages\":[],\"useMarkerImageFunction\":false,\"color\":\"#fe7569\",\"mapImageUrl\":\"\",\"xPosKeyName\":\"xPos\",\"yPosKeyName\":\"yPos\",\"posFunction\":\"return {x: origXPos, y: origYPos};\",\"markerOffsetX\":0.5,\"markerOffsetY\":1,\"showTooltip\":true,\"autocloseTooltip\":true,\"showTooltipAction\":\"click\",\"defaultCenterPosition\":\"0,0\",\"provider\":\"image-map\",\"fitMapBounds\":true,\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"polygonKeyName\":\"coordinates\",\"polygonOpacity\":0.5,\"polygonStrokeOpacity\":1,\"polygonStrokeWeight\":1,\"mapProvider\":\"HERE.normalDay\",\"draggableMarker\":true,\"editablePolygon\":true,\"mapPageSize\":16384},\"title\":\"Markers Placement - Image Map\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{\"tooltipAction\":[{\"name\":\"delete\",\"icon\":\"more_horiz\",\"type\":\"custom\",\"customFunction\":\"\\nvar entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id\\n });\\n\\nwidgetContext.map.setMarkerLocation(entityDatasource[0],\\n null,\\n null\\n ).subscribe(function succes() {\\n widgetContext.updateAliases();\\n });\",\"id\":\"c39f512a-21c6-6b06-3aa1-715262c6553d\"},{\"name\":\"delete_polygon\",\"icon\":\"more_horiz\",\"type\":\"customPretty\",\"customHtml\":\"<!--=======================================================================-->\\r\\n<!--===== There are two example templates: for edit and add entity =====-->\\r\\n<!--=======================================================================-->\\r\\n<!--======================== Edit entity example ========================-->\\r\\n<!--=======================================================================-->\\r\\n<!-- -->\\r\\n<!--<form #editEntityForm=\\\"ngForm\\\" [formGroup]=\\\"editEntityFormGroup\\\"-->\\r\\n<!-- (ngSubmit)=\\\"save()\\\" class=\\\"edit-entity-form\\\">-->\\r\\n<!-- <mat-toolbar fxLayout=\\\"row\\\" color=\\\"primary\\\">-->\\r\\n<!-- <h2>Edit {{entityType.toLowerCase()}} {{entityName}}</h2>-->\\r\\n<!-- <span fxFlex></span>-->\\r\\n<!-- <button mat-icon-button (click)=\\\"cancel()\\\" type=\\\"button\\\">-->\\r\\n<!-- <mat-icon class=\\\"material-icons\\\">close</mat-icon>-->\\r\\n<!-- </button>-->\\r\\n<!-- </mat-toolbar>-->\\r\\n<!-- <mat-progress-bar color=\\\"warn\\\" mode=\\\"indeterminate\\\" *ngIf=\\\"isLoading$ | async\\\">-->\\r\\n<!-- </mat-progress-bar>-->\\r\\n<!-- <div style=\\\"height: 4px;\\\" *ngIf=\\\"!(isLoading$ | async)\\\"></div>-->\\r\\n<!-- <div mat-dialog-content fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Entity Name</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"entityName\\\" required readonly=\\\"\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Entity Label</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"entityLabel\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Entity Type</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"entityType\\\" readonly>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Type</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"type\\\" readonly>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div formGroupName=\\\"attributes\\\" fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Latitude</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"latitude\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Longitude</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"longitude\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Address</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"address\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Owner</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"owner\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Integer Value</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"1\\\" matInput formControlName=\\\"number\\\">-->\\r\\n<!-- <mat-error *ngIf=\\\"editEntityFormGroup.get('attributes.number').hasError('pattern')\\\">-->\\r\\n<!-- Invalid integer value.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <div class=\\\"boolean-value-input\\\" fxLayout=\\\"column\\\" fxLayoutAlign=\\\"center start\\\" fxFlex>-->\\r\\n<!-- <label class=\\\"checkbox-label\\\">Boolean Value</label>-->\\r\\n<!-- <mat-checkbox formControlName=\\\"booleanValue\\\" style=\\\"margin-bottom: 40px;\\\">-->\\r\\n<!-- {{ (editEntityFormGroup.get('attributes.booleanValue').value ? \\\"value.true\\\" : \\\"value.false\\\") | translate }}-->\\r\\n<!-- </mat-checkbox>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div class=\\\"relations-list old-relations\\\">-->\\r\\n<!-- <div class=\\\"mat-body-1\\\" style=\\\"padding-bottom: 10px; color: rgba(0,0,0,0.57);\\\">Relations</div>-->\\r\\n<!-- <div class=\\\"body\\\" [fxShow]=\\\"oldRelations().length\\\">-->\\r\\n<!-- <div class=\\\"row\\\" fxLayout=\\\"row\\\" fxLayoutAlign=\\\"start center\\\" formArrayName=\\\"oldRelations\\\" -->\\r\\n<!-- *ngFor=\\\"let relation of oldRelations().controls; let i = index;\\\">-->\\r\\n<!-- <div [formGroupName]=\\\"i\\\" class=\\\"mat-elevation-z2\\\" fxFlex fxLayout=\\\"row\\\" style=\\\"padding: 5px 0 5px 5px;\\\">-->\\r\\n<!-- <div fxFlex fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field class=\\\"mat-block\\\" style=\\\"min-width: 100px;\\\">-->\\r\\n<!-- <mat-label>Direction</mat-label>-->\\r\\n<!-- <mat-select formControlName=\\\"direction\\\" name=\\\"direction\\\">-->\\r\\n<!-- <mat-option *ngFor=\\\"let direction of entitySearchDirection | keyvalue\\\" [value]=\\\"direction.value\\\">-->\\r\\n<!-- {{ (\\\"relation.search-direction.\\\" + direction.value) | translate}}-->\\r\\n<!-- </mat-option>-->\\r\\n<!-- </mat-select>-->\\r\\n<!-- <mat-error *ngIf=\\\"relation.get('direction').hasError('required')\\\">-->\\r\\n<!-- Relation direction is required.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <tb-relation-type-autocomplete-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"relationType\\\"-->\\r\\n<!-- required=\\\"true\\\">-->\\r\\n<!-- </tb-relation-type-autocomplete>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayout.xs=\\\"column\\\">-->\\r\\n<!-- <tb-entity-select-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- required=\\\"true\\\"-->\\r\\n<!-- formControlName=\\\"relatedEntity\\\">-->\\r\\n<!-- </tb-entity-select>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"column\\\" fxLayoutAlign=\\\"center center\\\">-->\\r\\n<!-- <button mat-icon-button color=\\\"primary\\\"-->\\r\\n<!-- aria-label=\\\"Remove\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- (click)=\\\"removeOldRelation(i)\\\"-->\\r\\n<!-- matTooltip=\\\"Remove relation\\\"-->\\r\\n<!-- matTooltipPosition=\\\"above\\\">-->\\r\\n<!-- <mat-icon>close</mat-icon>-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div class=\\\"relations-list\\\">-->\\r\\n<!-- <div class=\\\"mat-body-1\\\" style=\\\"padding-bottom: 10px; color: rgba(0,0,0,0.57);\\\">New Relations</div>-->\\r\\n<!-- <div class=\\\"body\\\" [fxShow]=\\\"relations().length\\\">-->\\r\\n<!-- <div class=\\\"row\\\" fxLayout=\\\"row\\\" fxLayoutAlign=\\\"start center\\\" formArrayName=\\\"relations\\\" *ngFor=\\\"let relation of relations().controls; let i = index;\\\">-->\\r\\n<!-- <div [formGroupName]=\\\"i\\\" class=\\\"mat-elevation-z2\\\" fxFlex fxLayout=\\\"row\\\" style=\\\"padding: 5px 0 5px 5px;\\\">-->\\r\\n<!-- <div fxFlex fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field class=\\\"mat-block\\\" style=\\\"min-width: 100px;\\\">-->\\r\\n<!-- <mat-label>Direction</mat-label>-->\\r\\n<!-- <mat-select formControlName=\\\"direction\\\" name=\\\"direction\\\">-->\\r\\n<!-- <mat-option *ngFor=\\\"let direction of entitySearchDirection | keyvalue\\\" [value]=\\\"direction.value\\\">-->\\r\\n<!-- {{ (\\\"relation.search-direction.\\\" + direction.value) | translate}}-->\\r\\n<!-- </mat-option>-->\\r\\n<!-- </mat-select>-->\\r\\n<!-- <mat-error *ngIf=\\\"relation.get('direction').hasError('required')\\\">-->\\r\\n<!-- Relation direction is required.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <tb-relation-type-autocomplete-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"relationType\\\"-->\\r\\n<!-- [required]=\\\"true\\\">-->\\r\\n<!-- </tb-relation-type-autocomplete>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayout.xs=\\\"column\\\">-->\\r\\n<!-- <tb-entity-select-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- [required]=\\\"true\\\"-->\\r\\n<!-- formControlName=\\\"relatedEntity\\\">-->\\r\\n<!-- </tb-entity-select>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"column\\\" fxLayoutAlign=\\\"center center\\\">-->\\r\\n<!-- <button mat-icon-button color=\\\"primary\\\"-->\\r\\n<!-- aria-label=\\\"Remove\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- (click)=\\\"removeRelation(i)\\\"-->\\r\\n<!-- matTooltip=\\\"Remove relation\\\"-->\\r\\n<!-- matTooltipPosition=\\\"above\\\">-->\\r\\n<!-- <mat-icon>close</mat-icon>-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div>-->\\r\\n<!-- <button mat-raised-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- (click)=\\\"addRelation()\\\"-->\\r\\n<!-- matTooltip=\\\"Add Relation\\\"-->\\r\\n<!-- matTooltipPosition=\\\"above\\\">-->\\r\\n<!-- Add-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div mat-dialog-actions fxLayout=\\\"row\\\" fxLayoutAlign=\\\"end center\\\">-->\\r\\n<!-- <button mat-button mat-raised-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"submit\\\"-->\\r\\n<!-- [disabled]=\\\"(isLoading$ | async) || editEntityForm.invalid || !editEntityForm.dirty\\\">-->\\r\\n<!-- Save-->\\r\\n<!-- </button>-->\\r\\n<!-- <button mat-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- [disabled]=\\\"(isLoading$ | async)\\\"-->\\r\\n<!-- (click)=\\\"cancel()\\\" cdkFocusInitial>-->\\r\\n<!-- Cancel-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!--</form>-->\\r\\n<!---->\\r\\n<!--========================================================================-->\\r\\n<!--========================= Add entity example =========================-->\\r\\n<!--========================================================================-->\\r\\n<!---->\\r\\n<!--<form #addEntityForm=\\\"ngForm\\\" [formGroup]=\\\"addEntityFormGroup\\\"-->\\r\\n<!-- (ngSubmit)=\\\"save()\\\" class=\\\"add-entity-form\\\">-->\\r\\n<!-- <mat-toolbar fxLayout=\\\"row\\\" color=\\\"primary\\\">-->\\r\\n<!-- <h2>Add entity</h2>-->\\r\\n<!-- <span fxFlex></span>-->\\r\\n<!-- <button mat-icon-button (click)=\\\"cancel()\\\" type=\\\"button\\\">-->\\r\\n<!-- <mat-icon class=\\\"material-icons\\\">close</mat-icon>-->\\r\\n<!-- </button>-->\\r\\n<!-- </mat-toolbar>-->\\r\\n<!-- <mat-progress-bar color=\\\"warn\\\" mode=\\\"indeterminate\\\" *ngIf=\\\"isLoading$ | async\\\">-->\\r\\n<!-- </mat-progress-bar>-->\\r\\n<!-- <div style=\\\"height: 4px;\\\" *ngIf=\\\"!(isLoading$ | async)\\\"></div>-->\\r\\n<!-- <div mat-dialog-content fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Entity Name</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"entityName\\\" required>-->\\r\\n<!-- <mat-error *ngIf=\\\"addEntityFormGroup.get('entityName').hasError('required')\\\">-->\\r\\n<!-- Entity name is required.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Entity Label</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"entityLabel\\\" >-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <tb-entity-type-select-->\\r\\n<!-- class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"entityType\\\"-->\\r\\n<!-- [showLabel]=\\\"true\\\"-->\\r\\n<!-- [allowedEntityTypes]=\\\"allowedEntityTypes\\\"-->\\r\\n<!-- ></tb-entity-type-select>-->\\r\\n<!-- <tb-entity-subtype-autocomplete-->\\r\\n<!-- fxFlex *ngIf=\\\"addEntityFormGroup.get('entityType').value == 'ASSET'\\\"-->\\r\\n<!-- class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"type\\\"-->\\r\\n<!-- [required]=\\\"true\\\"-->\\r\\n<!-- [entityType]=\\\"'ASSET'\\\"-->\\r\\n<!-- ></tb-entity-subtype-autocomplete>-->\\r\\n<!-- <tb-entity-subtype-autocomplete-->\\r\\n<!-- fxFlex *ngIf=\\\"addEntityFormGroup.get('entityType').value != 'ASSET'\\\"-->\\r\\n<!-- class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"type\\\"-->\\r\\n<!-- [required]=\\\"true\\\"-->\\r\\n<!-- [entityType]=\\\"'DEVICE'\\\"-->\\r\\n<!-- ></tb-entity-subtype-autocomplete>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div formGroupName=\\\"attributes\\\" fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Latitude</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"latitude\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Longitude</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"longitude\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Address</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"address\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Owner</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"owner\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Integer Value</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"1\\\" matInput formControlName=\\\"number\\\">-->\\r\\n<!-- <mat-error *ngIf=\\\"addEntityFormGroup.get('attributes.number').hasError('pattern')\\\">-->\\r\\n<!-- Invalid integer value.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <div class=\\\"boolean-value-input\\\" fxLayout=\\\"column\\\" fxLayoutAlign=\\\"center start\\\" fxFlex>-->\\r\\n<!-- <label class=\\\"checkbox-label\\\">Boolean Value</label>-->\\r\\n<!-- <mat-checkbox formControlName=\\\"booleanValue\\\" style=\\\"margin-bottom: 40px;\\\">-->\\r\\n<!-- {{ (addEntityFormGroup.get('attributes.booleanValue').value ? \\\"value.true\\\" : \\\"value.false\\\") | translate }}-->\\r\\n<!-- </mat-checkbox>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div class=\\\"relations-list\\\">-->\\r\\n<!-- <div class=\\\"mat-body-1\\\" style=\\\"padding-bottom: 10px; color: rgba(0,0,0,0.57);\\\">Relations</div>-->\\r\\n<!-- <div class=\\\"body\\\" [fxShow]=\\\"relations().length\\\">-->\\r\\n<!-- <div class=\\\"row\\\" fxLayout=\\\"row\\\" fxLayoutAlign=\\\"start center\\\" formArrayName=\\\"relations\\\" *ngFor=\\\"let relation of relations().controls; let i = index;\\\">-->\\r\\n<!-- <div [formGroupName]=\\\"i\\\" class=\\\"mat-elevation-z2\\\" fxFlex fxLayout=\\\"row\\\" style=\\\"padding: 5px 0 5px 5px;\\\">-->\\r\\n<!-- <div fxFlex fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field class=\\\"mat-block\\\" style=\\\"min-width: 100px;\\\">-->\\r\\n<!-- <mat-label>Direction</mat-label>-->\\r\\n<!-- <mat-select formControlName=\\\"direction\\\" name=\\\"direction\\\">-->\\r\\n<!-- <mat-option *ngFor=\\\"let direction of entitySearchDirection | keyvalue\\\" [value]=\\\"direction.value\\\">-->\\r\\n<!-- {{ (\\\"relation.search-direction.\\\" + direction.value) | translate}}-->\\r\\n<!-- </mat-option>-->\\r\\n<!-- </mat-select>-->\\r\\n<!-- <mat-error *ngIf=\\\"relation.get('direction').hasError('required')\\\">-->\\r\\n<!-- Relation direction is required.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <tb-relation-type-autocomplete-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"relationType\\\"-->\\r\\n<!-- [required]=\\\"true\\\">-->\\r\\n<!-- </tb-relation-type-autocomplete>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayout.xs=\\\"column\\\">-->\\r\\n<!-- <tb-entity-select-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- [required]=\\\"true\\\"-->\\r\\n<!-- formControlName=\\\"relatedEntity\\\">-->\\r\\n<!-- </tb-entity-select>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"column\\\" fxLayoutAlign=\\\"center center\\\">-->\\r\\n<!-- <button mat-icon-button color=\\\"primary\\\"-->\\r\\n<!-- aria-label=\\\"Remove\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- (click)=\\\"removeRelation(i)\\\"-->\\r\\n<!-- matTooltip=\\\"Remove relation\\\"-->\\r\\n<!-- matTooltipPosition=\\\"above\\\">-->\\r\\n<!-- <mat-icon>close</mat-icon>-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div>-->\\r\\n<!-- <button mat-raised-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- (click)=\\\"addRelation()\\\"-->\\r\\n<!-- matTooltip=\\\"Add Relation\\\"-->\\r\\n<!-- matTooltipPosition=\\\"above\\\">-->\\r\\n<!-- Add-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div mat-dialog-actions fxLayout=\\\"row\\\" fxLayoutAlign=\\\"end center\\\">-->\\r\\n<!-- <button mat-button mat-raised-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"submit\\\"-->\\r\\n<!-- [disabled]=\\\"(isLoading$ | async) || addEntityForm.invalid || !addEntityForm.dirty\\\">-->\\r\\n<!-- Create-->\\r\\n<!-- </button>-->\\r\\n<!-- <button mat-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- [disabled]=\\\"(isLoading$ | async)\\\"-->\\r\\n<!-- (click)=\\\"cancel()\\\" cdkFocusInitial>-->\\r\\n<!-- Cancel-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!--</form>-->\\r\\n\",\"customCss\":\"/*=======================================================================*/\\r\\n/*========== There are two examples: for edit and add entity ==========*/\\r\\n/*=======================================================================*/\\r\\n/*======================== Edit entity example ========================*/\\r\\n/*=======================================================================*/\\r\\n/*\\r\\n.edit-entity-form .boolean-value-input {\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.edit-entity-form .boolean-value-input .checkbox-label {\\r\\n margin-bottom: 8px;\\r\\n color: rgba(0,0,0,0.54);\\r\\n font-size: 12px;\\r\\n}\\r\\n\\r\\n.relations-list .header {\\r\\n padding-right: 5px;\\r\\n padding-bottom: 5px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .header .cell {\\r\\n padding-right: 5px;\\r\\n padding-left: 5px;\\r\\n font-size: 12px;\\r\\n font-weight: 700;\\r\\n color: rgba(0, 0, 0, .54);\\r\\n white-space: nowrap;\\r\\n}\\r\\n\\r\\n.relations-list .mat-form-field-infix {\\r\\n width: auto !important;\\r\\n}\\r\\n\\r\\n.relations-list .body {\\r\\n padding-right: 5px;\\r\\n padding-bottom: 15px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .row {\\r\\n padding-top: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .cell {\\r\\n padding-right: 5px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .md-button {\\r\\n margin: 0;\\r\\n}\\r\\n*/\\r\\n/*========================================================================*/\\r\\n/*========================= Add entity example =========================*/\\r\\n/*========================================================================*/\\r\\n/*\\r\\n.add-entity-form .boolean-value-input {\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.add-entity-form .boolean-value-input .checkbox-label {\\r\\n margin-bottom: 8px;\\r\\n color: rgba(0,0,0,0.54);\\r\\n font-size: 12px;\\r\\n}\\r\\n\\r\\n.relations-list .header {\\r\\n padding-right: 5px;\\r\\n padding-bottom: 5px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .header .cell {\\r\\n padding-right: 5px;\\r\\n padding-left: 5px;\\r\\n font-size: 12px;\\r\\n font-weight: 700;\\r\\n color: rgba(0, 0, 0, .54);\\r\\n white-space: nowrap;\\r\\n}\\r\\n\\r\\n.relations-list .mat-form-field-infix {\\r\\n width: auto !important;\\r\\n}\\r\\n\\r\\n.relations-list .body {\\r\\n padding-right: 5px;\\r\\n padding-bottom: 15px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .row {\\r\\n padding-top: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .cell {\\r\\n padding-right: 5px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .md-button {\\r\\n margin: 0;\\r\\n}\\r\\n*/\\r\\n\",\"customFunction\":\"\\nvar entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id\\n });\\n\\nwidgetContext.map.savePolygonLocation(entityDatasource[0],\\n null\\n ).subscribe(function succes() {\\n widgetContext.updateAliases();\\n });\",\"customResources\":[],\"id\":\"94bf5ffd-b526-c6c3-ae3b-ab42191217d9\"}]},\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"displayTimewindow\":true}"
342 358 }
343 359 },
344 360 {
345   - "alias": "update_server_location_attribute",
346   - "name": "Update server location attribute",
  361 + "alias": "web_camera_input",
  362 + "name": "Web Camera Input",
347 363 "descriptor": {
348 364 "type": "latest",
349 365 "sizeX": 7.5,
350 366 "sizeY": 3,
351 367 "resources": [],
352   - "templateHtml": "<div tb-toast toastTarget=\"{{ toastTargetId }}\" style=\"width: 100%; height: 100%;\">\n <form *ngIf=\"attributeUpdateFormGroup\"\n class=\"attribute-update-form\"\n [formGroup]=\"attributeUpdateFormGroup\"\n [ngClass]=\"{'small-width': smallWidthContainer}\"\n (ngSubmit)=\"updateAttribute()\">\n <div style=\"padding: 0 8px; margin: auto 0;\">\n <div class=\"attribute-update-form__grid\" [fxShow]=\"entityDetected && isValidParameter && dataKeyDetected\">\n <div class=\"grid__element\" \n [ngClass]=\"{'horizontal-alignment': isHorizontal && !changeAlignment}\">\n <mat-form-field class=\"mat-block\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? latLabel : '' }}</mat-label>\n <input matInput\n formControlName=\"currentLat\"\n required\n type=\"number\"\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n max=\"90\"\n min=\"-90\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentLat').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field>\n \n <mat-form-field class=\"mat-block\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? lngLabel : '' }}</mat-label>\n <input matInput\n formControlName=\"currentLng\"\n required\n type=\"number\"\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n max=\"180\"\n min=\"-180\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentLng').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field> \n </div>\n \n <div class=\"grid__element\">\n <button mat-button mat-icon-button class=\"getLocation\"\n type=\"button\"\n (click)=\"getCoordinate()\"\n *ngIf=\"settings.showGetLocation\"\n matTooltip=\"{{ 'widgets.input-widgets.get-location' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>my_location</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"applyChanges\"\n type=\"submit\"\n [disabled]=\"disableButton() || attributeUpdateFormGroup.invalid || attributeUpdateFormGroup.pristine\"\n matTooltip=\"{{ 'widgets.input-widgets.update-attribute' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>check</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"discardChanges\"\n type=\"button\"\n [disabled]=\"disableButton()\"\n (click)=\"discardChange()\"\n matTooltip=\"{{ 'widgets.input-widgets.discard-changes' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>close</mat-icon>\n </button>\n </div>\n </div>\n \n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\" [fxHide]=\"entityDetected\" >\n {{ 'widgets.input-widgets.no-entity-selected' | translate }}\n </div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !dataKeyDetected\">\n {{ 'widgets.input-widgets.no-attribute-selected' | translate }}\n </div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !isValidParameter\">\n {{ 'widgets.input-widgets.no-coordinate-specified' | translate }}\n </div>\n </div>\n </form>\n</div>",
353   - "templateCss": ".attribute-update-form {\n overflow: hidden;\n height: 100%;\n display: flex;\n flex-direction: column;\n}\n\n.attribute-update-form__grid {\n display: flex;\n}\n.grid__element:first-child {\n flex-direction: column;\n flex: 1;\n}\n\n.grid__element.horizontal-alignment {\n flex-direction: row;\n}\n\n.grid__element:last-child {\n align-items: center;\n margin-left: 7px;\n}\n.grid__element {\n display: flex;\n}\n\n.attribute-update-form .mat-button.mat-icon-button {\n width: 32px;\n min-width: 32px;\n height: 32px;\n min-height: 32px;\n padding: 0 !important;\n margin: 0;\n line-height: 20px;\n}\n\n.attribute-update-form .mat-button.getLocation {\n margin-right: 10px;\n}\n\n.attribute-update-form .mat-icon-button mat-icon {\n width: 20px;\n min-width: 20px;\n height: 20px;\n min-height: 20px;\n font-size: 20px;\n}\n\n.attribute-update-form mat-form-field{\n width: 100%;\n padding-right: 5px;\n}\n\n.attribute-update-form.small-width mat-form-field{\n width: 150px;\n}\n\n.tb-toast {\n font-size: 14px!important;\n}",
354   - "controllerScript": "let $scope;\r\nlet settings;\r\nlet attributeService;\r\nlet utils;\r\nlet translate;\r\n\r\nself.onInit = function() {\r\n self.ctx.ngZone.run(function() {\r\n init(); \r\n self.ctx.detectChanges(true);\r\n });\r\n};\r\n\r\n\r\nfunction init() {\r\n $scope = self.ctx.$scope;\r\n attributeService = $scope.$injector.get(self.ctx.servicesMap.get('attributeService'));\r\n utils = $scope.$injector.get(self.ctx.servicesMap.get('utils'));\r\n translate = $scope.$injector.get(self.ctx.servicesMap.get('translate'));\r\n $scope.toastTargetId = 'input-widget' + utils.guid();\r\n settings = utils.deepClone(self.ctx.settings) || {};\r\n \r\n settings.showLabel = utils.defaultValue(settings.showLabel, true);\r\n settings.showResultMessage = utils.defaultValue(settings.showResultMessage, true);\r\n settings.showGetLocation = utils.defaultValue(settings.showGetLocation, true);\r\n settings.enableHighAccuracy = utils.defaultValue(settings.enableHighAccuracy, false);\r\n $scope.settings = settings;\r\n $scope.isValidParameter = true;\r\n $scope.dataKeyDetected = false; \r\n\r\n $scope.isHorizontal = (settings.inputFieldsAlignment === 'row');\r\n $scope.requiredErrorMessage = utils.customTranslation(settings.requiredErrorMessage, settings.requiredErrorMessage) || translate.instant('widgets.input-widgets.entity-coordinate-required');\r\n $scope.latLabel = utils.customTranslation(settings.latLabel, settings.latLabel) || translate.instant('widgets.input-widgets.latitude');\r\n $scope.lngLabel = utils.customTranslation(settings.lngLabel, settings.lngLabel) || translate.instant('widgets.input-widgets.longitude');\r\n\r\n $scope.attributeUpdateFormGroup = $scope.fb.group(\r\n {currentLat: [undefined, [$scope.validators.required,\r\n $scope.validators.min(-90),\r\n $scope.validators.max(90)]],\r\n currentLng: [undefined, [$scope.validators.required,\r\n $scope.validators.min(-180),\r\n $scope.validators.max(180)]]}\r\n );\r\n\r\n if (self.ctx.datasources && self.ctx.datasources.length) {\r\n var datasource = self.ctx.datasources[0];\r\n if (datasource.type === 'entity') {\r\n if (datasource.entityType && datasource.entityId) {\r\n $scope.entityName = datasource.entityName;\r\n if (settings.widgetTitle && settings.widgetTitle.length) {\r\n $scope.titleTemplate = utils.customTranslation(settings.widgetTitle, settings.widgetTitle);\r\n } else {\r\n $scope.titleTemplate = self.ctx.widgetConfig.title;\r\n }\r\n\r\n $scope.entityDetected = true;\r\n }\r\n }\r\n if (datasource.dataKeys.length > 1) {\r\n $scope.dataKeyDetected = true;\r\n for (let i = 0; i < datasource.dataKeys.length; i++) {\r\n if (datasource.dataKeys[i].type != \"attribute\"){\r\n $scope.isValidParameter = false;\r\n }\r\n if (datasource.dataKeys[i].name !== settings.latKeyName && datasource.dataKeys[i].name !== settings.lngKeyName){\r\n $scope.dataKeyDetected = false;\r\n }\r\n }\r\n }\r\n }\r\n\r\n self.ctx.widgetTitle = utils.createLabelFromDatasource(self.ctx.datasources[0], $scope.titleTemplate);\r\n\r\n $scope.updateAttribute = function () {\r\n $scope.isFocused = false;\r\n if ($scope.entityDetected) {\r\n var datasource = self.ctx.datasources[0];\r\n\r\n attributeService.saveEntityAttributes(\r\n datasource.entity.id,\r\n 'SERVER_SCOPE',\r\n [\r\n {\r\n key: settings.latKeyName,\r\n value: $scope.attributeUpdateFormGroup.get('currentLat').value\r\n },{\r\n key: settings.lngKeyName,\r\n value: $scope.attributeUpdateFormGroup.get('currentLng').value\r\n }\r\n ]\r\n ).subscribe(\r\n function success() {\r\n $scope.originalLat = $scope.attributeUpdateFormGroup.get('currentLat').value;\r\n $scope.originalLng = $scope.attributeUpdateFormGroup.get('currentLng').value;\r\n if (settings.showResultMessage) {\r\n $scope.showSuccessToast(translate.instant('widgets.input-widgets.update-successful'), 1000, 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n },\r\n function fail() {\r\n if (settings.showResultMessage) {\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.update-failed'), 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n }\r\n );\r\n }\r\n };\r\n\r\n $scope.changeFocus = function () {\r\n if ($scope.attributeUpdateFormGroup.get('currentLat').value === $scope.originalLat && $scope.attributeUpdateFormGroup.get('currentLng').value === $scope.originalLng) {\r\n $scope.isFocused = false;\r\n }\r\n };\r\n \r\n $scope.discardChange = function() {\r\n $scope.attributeUpdateFormGroup.setValue({\r\n 'currentLat': $scope.originalLat,\r\n 'currentLng': $scope.originalLng\r\n });\r\n $scope.isFocused = false;\r\n $scope.attributeUpdateFormGroup.markAsPristine();\r\n self.onDataUpdated();\r\n };\r\n \r\n $scope.disableButton = function () {\r\n return $scope.attributeUpdateFormGroup.get('currentLat').value === $scope.originalLat && $scope.attributeUpdateFormGroup.get('currentLng').value === $scope.originalLng || $scope.currentLng === null || $scope.currentLat === null;\r\n };\r\n \r\n $scope.getCoordinate = function() {\r\n if (navigator.geolocation) {\r\n navigator.geolocation.getCurrentPosition(showPosition, function (){\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.blocked-location'), \r\n 'bottom', 'left', $scope.toastTargetId);\r\n }, {\r\n enableHighAccuracy: settings.enableHighAccuracy\r\n });\r\n } else {\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.no-support-geolocation'), 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n };\r\n \r\n function showPosition(position) {\r\n $scope.attributeUpdateFormGroup.setValue({\r\n currentLat: correctValue(position.coords.latitude),\r\n currentLng: correctValue(position.coords.longitude)\r\n });\r\n $scope.attributeUpdateFormGroup.markAsDirty();\r\n $scope.isFocused = true;\r\n }\r\n \r\n self.onResize();\r\n}\r\n\r\nself.onDataUpdated = function() {\r\n try {\r\n if ($scope.dataKeyDetected) {\r\n if (!$scope.isFocused) {\r\n for(let i = 0; i < self.typeParameters().maxDataKeys; i++){\r\n if(self.ctx.data[i].dataKey.name === self.ctx.settings.latKeyName && $scope.attributeUpdateFormGroup.get('currentLat').pristine){\r\n $scope.originalLat = self.ctx.data[i].data[0][1];\r\n $scope.attributeUpdateFormGroup.get('currentLat').patchValue(correctValue($scope.originalLat));\r\n } else if(self.ctx.data[i].dataKey.name === self.ctx.settings.lngKeyName && $scope.attributeUpdateFormGroup.get('currentLng').pristine){\r\n $scope.originalLng = self.ctx.data[i].data[0][1];\r\n $scope.attributeUpdateFormGroup.get('currentLng').patchValue(correctValue($scope.originalLng));\r\n }\r\n }\r\n self.ctx.detectChanges();\r\n }\r\n }\r\n } catch (e) {\r\n console.log(e);\r\n }\r\n};\r\n\r\nfunction correctValue(value) {\r\n if (typeof value !== \"number\") {\r\n return 0;\r\n }\r\n return value;\r\n}\r\n\r\nself.onResize = function() {\r\n $scope.smallWidthContainer = (self.ctx.$container && self.ctx.$container[0].offsetWidth < 320);\r\n $scope.changeAlignment = ($scope.isHorizontal && self.ctx.$container && self.ctx.$container[0].offsetWidth < 480);\r\n self.ctx.detectChanges();\r\n};\r\n\r\nself.typeParameters = function() {\r\n return {\r\n maxDatasources: 1,\r\n maxDataKeys: 2,\r\n singleEntity: true\r\n };\r\n};\r\n\r\nself.onDestroy = function() {\r\n\r\n};",
355   - "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"widgetTitle\": {\n \"title\": \"Widget title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"latKeyName\": {\n \"title\": \"Latitude key name\",\n \"type\": \"string\",\n \"default\": \"latitude\"\n },\n \"lngKeyName\": {\n \"title\": \"Longitude key name\",\n \"type\": \"string\",\n \"default\": \"longitude\"\n },\n \"showLabel\": {\n \"title\": \"Show label\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"latLabel\": {\n \"title\": \"Label for latitude\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"lngLabel\": {\n \"title\": \"Label for longitude\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"requiredErrorMessage\": {\n \"title\": \"'Required' error message\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"showResultMessage\": {\n \"title\": \"Show result message\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableHighAccuracy\": {\n \"title\": \"Use high accuracy\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"showGetLocation\": {\n \"title\": \"Show button 'Get current location'\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"inputFieldsAlignment\": {\n \"title\": \"Input fields alignment\",\n \"type\": \"string\",\n \"default\": \"column\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"widgetTitle\",\n \"latKeyName\",\n \"lngKeyName\",\n \"enableHighAccuracy\",\n \"showGetLocation\",\n \"showResultMessage\",\n {\n \"key\": \"inputFieldsAlignment\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"column\",\n \"label\": \"Column (default)\"\n },\n {\n \"value\": \"row\",\n \"label\": \"Row\"\n }\n ]\n },\n \"showLabel\",\n \"latLabel\",\n \"lngLabel\",\n \"requiredErrorMessage\"\n ]\n}",
  368 + "templateHtml": "<tb-web-camera-widget \n [ctx]=\"ctx\">\n</tb-web-camera-widget>",
  369 + "templateCss": "",
  370 + "controllerScript": "self.onInit = function() {\n}\n\nself.onDataUpdated = function() {\n self.ctx.$scope.webCameraInputWidget.onDataUpdated();\n}\n\nself.typeParameters = function() {\n return {\n maxDatasources: 1,\n maxDataKeys: 1,\n singleEntity: true\n }\n}\n\nself.onDestroy = function() {\n}\n",
  371 + "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Web Camera\",\n \"properties\": {\n \"widgetTitle\": {\n \"title\": \"Widget title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"imageFormat\": {\n \"title\": \"Image Format\",\n \"type\": \"string\",\n \"default\": \"image/png\"\n },\n \"imageQuality\":{\n \"title\":\"Image quality that use lossy compression such as jpeg and webp\",\n \"type\":\"number\",\n \"default\": 0.92,\n \"min\": 0,\n \"max\": 1\n },\n \"maxWidth\": {\n \"title\": \"The maximal image width\",\n \"type\": \"number\",\n \"default\": 640\n }, \n \"maxHeight\": {\n \"title\": \"The maximal image heigth\",\n \"type\": \"number\",\n \"default\": 480\n }\n },\n \"required\": []\n },\n \"form\": [\n \"widgetTitle\",\n {\n \"key\": \"imageFormat\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"image/jpeg\",\n \"label\": \"JPEG\"\n },\n {\n \"value\": \"image/png\",\n \"label\": \"PNG\"\n },\n {\n \"value\": \"image/webp\",\n \"label\": \"WEBP\"\n }\n ]\n },\n \"imageQuality\",\n \"maxWidth\",\n \"maxHeight\"\n ]\n}",
356 372 "dataKeySettingsSchema": "{}\n",
357   - "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Update server location attribute\",\"dropShadow\":true,\"enableFullscreen\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
  373 + "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Web Camera Input\",\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"dropShadow\":true,\"enableFullscreen\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"displayTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
358 374 }
359 375 },
360 376 {
... ... @@ -370,39 +386,23 @@
370 386 "controllerScript": "self.onInit = function() {\n self.ctx.map = new TbMapWidgetV2('google-map', false, self.ctx, null, true);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbMapWidgetV2.settingsSchema('google-map');\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbMapWidgetV2.dataKeySettingsSchema('google-map');\n}\n\nself.actionSources = function() {\n return TbMapWidgetV2.actionSources();\n}\n\nself.onDestroy = function() {\n}\n\nself.typeParameters = function() {\n return {\n hasDataPageLink: true\n };\n}",
371 387 "settingsSchema": "{}",
372 388 "dataKeySettingsSchema": "{}\n",
373   - "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.05427416942713381,\"funcBody\":\"var value = prevValue || 15.833293;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.680594833308841,\"funcBody\":\"var value = prevValue || -90.454350;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]},{\"type\":\"function\",\"name\":\"Second point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.05012157428742059,\"funcBody\":\"var value = prevValue || 14.450463;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.6742359401617628,\"funcBody\":\"var value = prevValue || -84.845334;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"fitMapBounds\":true,\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"label\":\"${entityName}\",\"tooltipPattern\":\"<b>${entityName}</b><br/><br/><b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><br/><link-act name='delete'>Delete</link-act>\",\"markerImageSize\":34,\"gmDefaultMapType\":\"roadmap\",\"gmApiKey\":\"AIzaSyDoEx2kaGz3PxwbI9T7ccTSg5xjdw8Nw8Q\",\"useColorFunction\":false,\"markerImages\":[],\"useMarkerImageFunction\":false,\"colorFunction\":\"\\n\",\"color\":\"#fe7569\",\"showTooltip\":true,\"autocloseTooltip\":true,\"defaultCenterPosition\":\"0,0\",\"showTooltipAction\":\"click\",\"polygonKeyName\":\"coordinates\",\"polygonOpacity\":0.5,\"polygonStrokeOpacity\":1,\"polygonStrokeWeight\":1,\"zoomOnClick\":true,\"defaultZoomLevel\":5,\"provider\":\"google-map\",\"showCoverageOnHover\":true,\"animate\":true,\"maxClusterRadius\":80,\"removeOutsideVisibleBounds\":true,\"mapProvider\":\"HERE.normalDay\",\"draggableMarker\":true},\"title\":\"Markers Placement - Google Maps\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{\"tooltipAction\":[{\"id\":\"8d3c0156-0a14-7a6f-0ddd-0ec16b9ffc91\",\"name\":\"delete\",\"icon\":\"more_horiz\",\"type\":\"custom\",\"customFunction\":\"var $rootScope = widgetContext.$scope.$injector.get('$rootScope');\\nvar entityDatasource = widgetContext.map.subscription.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id\\n });\\n\\nwidgetContext.map.saveMarkerLocation(entityDatasource[0],\\n widgetContext.map.locations[0], {\\n \\\"lat\\\": null,\\n \\\"lng\\\": null\\n }).then(function succes() {\\n $rootScope.$broadcast('widgetForceReInit');\\n });\"}]},\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"displayTimewindow\":true}"
374   - }
375   - },
376   - {
377   - "alias": "update_shared_location_attribute",
378   - "name": "Update shared location attribute",
379   - "descriptor": {
380   - "type": "latest",
381   - "sizeX": 7.5,
382   - "sizeY": 3,
383   - "resources": [],
384   - "templateHtml": "<div tb-toast toastTarget=\"{{ toastTargetId }}\" style=\"width: 100%; height: 100%;\">\n <form *ngIf=\"attributeUpdateFormGroup\"\n class=\"attribute-update-form\"\n [formGroup]=\"attributeUpdateFormGroup\"\n [ngClass]=\"{'small-width': smallWidthContainer}\"\n (ngSubmit)=\"updateAttribute()\">\n <div style=\"padding: 0 8px; margin: auto 0;\">\n <div class=\"attribute-update-form__grid\" [fxShow]=\"entityDetected && isValidParameter && dataKeyDetected\">\n <div class=\"grid__element\" \n [ngClass]=\"{'horizontal-alignment': isHorizontal && !changeAlignment}\">\n <mat-form-field class=\"mat-block\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? latLabel : '' }}</mat-label>\n <input matInput\n formControlName=\"currentLat\"\n required\n type=\"number\"\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n max=\"90\"\n min=\"-90\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentLat').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field>\n \n <mat-form-field class=\"mat-block\"\n floatLabel=\"{{settings.showLabel ? 'auto' : 'always'}}\"\n [hideRequiredMarker]=\"!settings.showLabel\">\n <mat-label>{{ settings.showLabel ? lngLabel : '' }}</mat-label>\n <input matInput\n formControlName=\"currentLng\"\n required\n type=\"number\"\n (focus)=\"isFocused = true\"\n (blur)=\"changeFocus()\"\n max=\"180\"\n min=\"-180\"/>\n <mat-error *ngIf=\"attributeUpdateFormGroup.get('currentLng').hasError('required')\">\n {{requiredErrorMessage}}\n </mat-error>\n </mat-form-field> \n </div>\n \n <div class=\"grid__element\">\n <button mat-button mat-icon-button class=\"getLocation\"\n type=\"button\"\n (click)=\"getCoordinate()\"\n *ngIf=\"settings.showGetLocation\"\n matTooltip=\"{{ 'widgets.input-widgets.get-location' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>my_location</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"applyChanges\"\n type=\"submit\"\n [disabled]=\"disableButton() || attributeUpdateFormGroup.invalid || attributeUpdateFormGroup.pristine\"\n matTooltip=\"{{ 'widgets.input-widgets.update-attribute' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>check</mat-icon>\n </button>\n <button mat-button mat-icon-button class=\"discardChanges\"\n type=\"button\"\n [disabled]=\"disableButton()\"\n (click)=\"discardChange()\"\n matTooltip=\"{{ 'widgets.input-widgets.discard-changes' | translate }}\"\n matTooltipPosition=\"above\">\n <mat-icon>close</mat-icon>\n </button>\n </div>\n </div>\n \n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\" [fxHide]=\"entityDetected\" [innerHtml]=\"message\"></div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !dataKeyDetected\">\n {{ 'widgets.input-widgets.no-attribute-selected' | translate }}\n </div>\n <div style=\"text-align: center; font-size: 18px; color: #a0a0a0;\"\n [fxShow]=\"entityDetected && !isValidParameter\">\n {{ 'widgets.input-widgets.no-coordinate-specified' | translate }}\n </div>\n </div>\n </form>\n</div>",
385   - "templateCss": ".attribute-update-form {\n overflow: hidden;\n height: 100%;\n display: flex;\n flex-direction: column;\n}\n\n.attribute-update-form__grid {\n display: flex;\n}\n.grid__element:first-child {\n flex-direction: column;\n flex: 1;\n}\n\n.grid__element.horizontal-alignment {\n flex-direction: row;\n}\n\n.grid__element:last-child {\n align-items: center;\n margin-left: 7px;\n}\n.grid__element {\n display: flex;\n}\n\n.attribute-update-form .mat-button.mat-icon-button {\n margin: 0;\n}\n\n.attribute-update-form .mat-button.mat-icon-button {\n width: 32px;\n min-width: 32px;\n height: 32px;\n min-height: 32px;\n padding: 0 !important;\n margin: 0;\n line-height: 20px;\n}\n\n.attribute-update-form .mat-button.getLocation {\n margin-right: 10px;\n}\n\n.attribute-update-form .mat-icon-button mat-icon {\n width: 20px;\n min-width: 20px;\n height: 20px;\n min-height: 20px;\n font-size: 20px;\n}\n\n.attribute-update-form mat-form-field{\n width: 100%;\n padding-right: 5px;\n}\n\n.attribute-update-form.small-width mat-form-field{\n width: 150px;\n}\n\n.tb-toast {\n font-size: 14px!important;\n}",
386   - "controllerScript": "let $scope;\r\nlet settings;\r\nlet attributeService;\r\nlet utils;\r\nlet translate;\r\n\r\nself.onInit = function() {\r\n self.ctx.ngZone.run(function() {\r\n init(); \r\n self.ctx.detectChanges(true);\r\n });\r\n};\r\n\r\n\r\nfunction init() {\r\n $scope = self.ctx.$scope;\r\n attributeService = $scope.$injector.get(self.ctx.servicesMap.get('attributeService'));\r\n utils = $scope.$injector.get(self.ctx.servicesMap.get('utils'));\r\n translate = $scope.$injector.get(self.ctx.servicesMap.get('translate'));\r\n $scope.toastTargetId = 'input-widget' + utils.guid();\r\n settings = utils.deepClone(self.ctx.settings) || {};\r\n \r\n settings.showLabel = utils.defaultValue(settings.showLabel, true);\r\n settings.showResultMessage = utils.defaultValue(settings.showResultMessage, true);\r\n settings.showGetLocation = utils.defaultValue(settings.showGetLocation, true);\r\n settings.enableHighAccuracy = utils.defaultValue(settings.enableHighAccuracy, false);\r\n $scope.settings = settings;\r\n $scope.isValidParameter = true;\r\n $scope.dataKeyDetected = false; \r\n $scope.message = translate.instant('widgets.input-widgets.no-entity-selected');\r\n\r\n $scope.isHorizontal = (settings.inputFieldsAlignment === 'row');\r\n $scope.requiredErrorMessage = utils.customTranslation(settings.requiredErrorMessage, settings.requiredErrorMessage) || translate.instant('widgets.input-widgets.entity-coordinate-required');\r\n $scope.latLabel = utils.customTranslation(settings.latLabel, settings.latLabel) || translate.instant('widgets.input-widgets.latitude');\r\n $scope.lngLabel = utils.customTranslation(settings.lngLabel, settings.lngLabel) || translate.instant('widgets.input-widgets.longitude');\r\n\r\n $scope.attributeUpdateFormGroup = $scope.fb.group(\r\n {currentLat: [undefined, [$scope.validators.required,\r\n $scope.validators.min(-90),\r\n $scope.validators.max(90)]],\r\n currentLng: [undefined, [$scope.validators.required,\r\n $scope.validators.min(-180),\r\n $scope.validators.max(180)]]}\r\n );\r\n\r\n if (self.ctx.datasources && self.ctx.datasources.length) {\r\n var datasource = self.ctx.datasources[0];\r\n if (datasource.type === 'entity') {\r\n if (datasource.entityType === 'DEVICE') {\r\n if (datasource.entityType && datasource.entityId) {\r\n $scope.entityName = datasource.entityName;\r\n if (settings.widgetTitle && settings.widgetTitle.length) {\r\n $scope.titleTemplate = utils.customTranslation(settings.widgetTitle, settings.widgetTitle);\r\n } else {\r\n $scope.titleTemplate = self.ctx.widgetConfig.title;\r\n }\r\n \r\n $scope.entityDetected = true;\r\n }\r\n } else {\r\n $scope.message = translate.instant('widgets.input-widgets.not-allowed-entity');\r\n }\r\n }\r\n if (datasource.dataKeys.length > 1) {\r\n $scope.dataKeyDetected = true;\r\n for (let i = 0; i < datasource.dataKeys.length; i++) {\r\n if (datasource.dataKeys[i].type != \"attribute\"){\r\n $scope.isValidParameter = false;\r\n }\r\n if (datasource.dataKeys[i].name !== settings.latKeyName && datasource.dataKeys[i].name !== settings.lngKeyName){\r\n $scope.dataKeyDetected = false;\r\n }\r\n }\r\n }\r\n }\r\n\r\n self.ctx.widgetTitle = utils.createLabelFromDatasource(self.ctx.datasources[0], $scope.titleTemplate);\r\n\r\n $scope.updateAttribute = function () {\r\n $scope.isFocused = false;\r\n if ($scope.entityDetected) {\r\n var datasource = self.ctx.datasources[0];\r\n\r\n attributeService.saveEntityAttributes(\r\n datasource.entity.id,\r\n 'SHARED_SCOPE',\r\n [\r\n {\r\n key: settings.latKeyName,\r\n value: $scope.attributeUpdateFormGroup.get('currentLat').value\r\n },{\r\n key: settings.lngKeyName,\r\n value: $scope.attributeUpdateFormGroup.get('currentLng').value\r\n }\r\n ]\r\n ).subscribe(\r\n function success() {\r\n $scope.originalLat = $scope.attributeUpdateFormGroup.get('currentLat').value;\r\n $scope.originalLng = $scope.attributeUpdateFormGroup.get('currentLng').value;\r\n if (settings.showResultMessage) {\r\n $scope.showSuccessToast(translate.instant('widgets.input-widgets.update-successful'), 1000, 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n },\r\n function fail() {\r\n if (settings.showResultMessage) {\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.update-failed'), 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n }\r\n );\r\n }\r\n };\r\n\r\n $scope.changeFocus = function () {\r\n if ($scope.attributeUpdateFormGroup.get('currentLat').value === $scope.originalLat && $scope.attributeUpdateFormGroup.get('currentLng').value === $scope.originalLng) {\r\n $scope.isFocused = false;\r\n }\r\n };\r\n \r\n $scope.discardChange = function() {\r\n $scope.attributeUpdateFormGroup.setValue({\r\n 'currentLat': $scope.originalLat,\r\n 'currentLng': $scope.originalLng\r\n });\r\n $scope.isFocused = false;\r\n $scope.attributeUpdateFormGroup.markAsPristine();\r\n self.onDataUpdated();\r\n };\r\n \r\n $scope.disableButton = function () {\r\n return $scope.attributeUpdateFormGroup.get('currentLat').value === $scope.originalLat && $scope.attributeUpdateFormGroup.get('currentLng').value === $scope.originalLng || $scope.currentLng === null || $scope.currentLat === null;\r\n };\r\n \r\n $scope.getCoordinate = function() {\r\n if (navigator.geolocation) {\r\n navigator.geolocation.getCurrentPosition(showPosition, function (){\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.blocked-location'), \r\n 'bottom', 'left', $scope.toastTargetId);\r\n }, {\r\n enableHighAccuracy: settings.enableHighAccuracy\r\n });\r\n } else {\r\n $scope.showErrorToast(translate.instant('widgets.input-widgets.no-support-geolocation'), 'bottom', 'left', $scope.toastTargetId);\r\n }\r\n };\r\n \r\n function showPosition(position) {\r\n $scope.attributeUpdateFormGroup.setValue({\r\n currentLat: correctValue(position.coords.latitude),\r\n currentLng: correctValue(position.coords.longitude)\r\n });\r\n $scope.attributeUpdateFormGroup.markAsDirty();\r\n $scope.isFocused = true;\r\n }\r\n \r\n self.onResize();\r\n}\r\n\r\nself.onDataUpdated = function() {\r\n try {\r\n if ($scope.dataKeyDetected) {\r\n if (!$scope.isFocused) {\r\n for(let i = 0; i < self.typeParameters().maxDataKeys; i++){\r\n if(self.ctx.data[i].dataKey.name === self.ctx.settings.latKeyName && $scope.attributeUpdateFormGroup.get('currentLat').pristine){\r\n $scope.originalLat = self.ctx.data[i].data[0][1];\r\n $scope.attributeUpdateFormGroup.get('currentLat').patchValue(correctValue($scope.originalLat));\r\n } else if(self.ctx.data[i].dataKey.name === self.ctx.settings.lngKeyName && $scope.attributeUpdateFormGroup.get('currentLng').pristine){\r\n $scope.originalLng = self.ctx.data[i].data[0][1];\r\n $scope.attributeUpdateFormGroup.get('currentLng').patchValue(correctValue($scope.originalLng));\r\n }\r\n }\r\n self.ctx.detectChanges();\r\n }\r\n }\r\n } catch (e) {\r\n console.log(e);\r\n }\r\n};\r\n\r\nfunction correctValue(value) {\r\n if (typeof value !== \"number\") {\r\n return 0;\r\n }\r\n return value;\r\n}\r\n\r\nself.onResize = function() {\r\n $scope.smallWidthContainer = (self.ctx.$container && self.ctx.$container[0].offsetWidth < 320);\r\n $scope.changeAlignment = ($scope.isHorizontal && self.ctx.$container && self.ctx.$container[0].offsetWidth < 480);\r\n self.ctx.detectChanges();\r\n};\r\n\r\nself.typeParameters = function() {\r\n return {\r\n maxDatasources: 1,\r\n maxDataKeys: 2,\r\n singleEntity: true\r\n };\r\n};\r\n\r\nself.onDestroy = function() {\r\n\r\n};",
387   - "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"widgetTitle\": {\n \"title\": \"Widget title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"latKeyName\": {\n \"title\": \"Latitude key name\",\n \"type\": \"string\",\n \"default\": \"latitude\"\n },\n \"lngKeyName\": {\n \"title\": \"Longitude key name\",\n \"type\": \"string\",\n \"default\": \"longitude\"\n },\n \"showLabel\": {\n \"title\": \"Show label\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"latLabel\": {\n \"title\": \"Label for latitude\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"lngLabel\": {\n \"title\": \"Label for longitude\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"requiredErrorMessage\": {\n \"title\": \"'Required' error message\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"showResultMessage\": {\n \"title\": \"Show result message\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableHighAccuracy\": {\n \"title\": \"Use high accuracy\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"showGetLocation\": {\n \"title\": \"Show button 'Get current location'\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"inputFieldsAlignment\": {\n \"title\": \"Input fields alignment\",\n \"type\": \"string\",\n \"default\": \"column\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"widgetTitle\",\n \"latKeyName\",\n \"lngKeyName\",\n \"enableHighAccuracy\",\n \"showGetLocation\",\n \"showResultMessage\",\n {\n \"key\": \"inputFieldsAlignment\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"column\",\n \"label\": \"Column (default)\"\n },\n {\n \"value\": \"row\",\n \"label\": \"Row\"\n }\n ]\n },\n \"showLabel\",\n \"latLabel\",\n \"lngLabel\",\n \"requiredErrorMessage\"\n ]\n}",
388   - "dataKeySettingsSchema": "{}\n",
389   - "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Update shared location attribute\",\"dropShadow\":true,\"enableFullscreen\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
  389 + "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.05427416942713381,\"funcBody\":\"var value = prevValue || 15.833293;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.680594833308841,\"funcBody\":\"var value = prevValue || -90.454350;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]},{\"type\":\"function\",\"name\":\"Second point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.05012157428742059,\"funcBody\":\"var value = prevValue || 14.450463;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.6742359401617628,\"funcBody\":\"var value = prevValue || -84.845334;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"fitMapBounds\":true,\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"label\":\"${entityName}\",\"tooltipPattern\":\"<b>${entityName}</b><br/><br/><b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><br/><link-act name='delete'>Delete</link-act>\",\"markerImageSize\":34,\"gmDefaultMapType\":\"roadmap\",\"gmApiKey\":\"AIzaSyDoEx2kaGz3PxwbI9T7ccTSg5xjdw8Nw8Q\",\"useColorFunction\":false,\"markerImages\":[],\"useMarkerImageFunction\":false,\"colorFunction\":\"\\n\",\"color\":\"#fe7569\",\"showTooltip\":true,\"autocloseTooltip\":true,\"defaultCenterPosition\":\"0,0\",\"showTooltipAction\":\"click\",\"polygonKeyName\":\"coordinates\",\"polygonOpacity\":0.5,\"polygonStrokeOpacity\":1,\"polygonStrokeWeight\":1,\"zoomOnClick\":true,\"defaultZoomLevel\":5,\"provider\":\"google-map\",\"showCoverageOnHover\":true,\"animate\":true,\"maxClusterRadius\":80,\"removeOutsideVisibleBounds\":true,\"mapProvider\":\"HERE.normalDay\",\"draggableMarker\":true,\"editablePolygon\":true,\"mapPageSize\":16384},\"title\":\"Markers Placement - Google Maps\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{\"tooltipAction\":[{\"name\":\"delete\",\"icon\":\"more_horiz\",\"type\":\"custom\",\"customFunction\":\"\\nvar entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id\\n });\\n\\nwidgetContext.map.setMarkerLocation(entityDatasource[0],\\n null,\\n null\\n ).subscribe(function succes() {\\n widgetContext.updateAliases();\\n });\",\"id\":\"8d3c0156-0a14-7a6f-0ddd-0ec16b9ffc91\"},{\"name\":\"delete_polygon\",\"icon\":\"more_horiz\",\"type\":\"customPretty\",\"customHtml\":\"<!--=======================================================================-->\\r\\n<!--===== There are two example templates: for edit and add entity =====-->\\r\\n<!--=======================================================================-->\\r\\n<!--======================== Edit entity example ========================-->\\r\\n<!--=======================================================================-->\\r\\n<!-- -->\\r\\n<!--<form #editEntityForm=\\\"ngForm\\\" [formGroup]=\\\"editEntityFormGroup\\\"-->\\r\\n<!-- (ngSubmit)=\\\"save()\\\" class=\\\"edit-entity-form\\\">-->\\r\\n<!-- <mat-toolbar fxLayout=\\\"row\\\" color=\\\"primary\\\">-->\\r\\n<!-- <h2>Edit {{entityType.toLowerCase()}} {{entityName}}</h2>-->\\r\\n<!-- <span fxFlex></span>-->\\r\\n<!-- <button mat-icon-button (click)=\\\"cancel()\\\" type=\\\"button\\\">-->\\r\\n<!-- <mat-icon class=\\\"material-icons\\\">close</mat-icon>-->\\r\\n<!-- </button>-->\\r\\n<!-- </mat-toolbar>-->\\r\\n<!-- <mat-progress-bar color=\\\"warn\\\" mode=\\\"indeterminate\\\" *ngIf=\\\"isLoading$ | async\\\">-->\\r\\n<!-- </mat-progress-bar>-->\\r\\n<!-- <div style=\\\"height: 4px;\\\" *ngIf=\\\"!(isLoading$ | async)\\\"></div>-->\\r\\n<!-- <div mat-dialog-content fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Entity Name</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"entityName\\\" required readonly=\\\"\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Entity Label</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"entityLabel\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Entity Type</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"entityType\\\" readonly>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Type</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"type\\\" readonly>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div formGroupName=\\\"attributes\\\" fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Latitude</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"latitude\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Longitude</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"longitude\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Address</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"address\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Owner</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"owner\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Integer Value</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"1\\\" matInput formControlName=\\\"number\\\">-->\\r\\n<!-- <mat-error *ngIf=\\\"editEntityFormGroup.get('attributes.number').hasError('pattern')\\\">-->\\r\\n<!-- Invalid integer value.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <div class=\\\"boolean-value-input\\\" fxLayout=\\\"column\\\" fxLayoutAlign=\\\"center start\\\" fxFlex>-->\\r\\n<!-- <label class=\\\"checkbox-label\\\">Boolean Value</label>-->\\r\\n<!-- <mat-checkbox formControlName=\\\"booleanValue\\\" style=\\\"margin-bottom: 40px;\\\">-->\\r\\n<!-- {{ (editEntityFormGroup.get('attributes.booleanValue').value ? \\\"value.true\\\" : \\\"value.false\\\") | translate }}-->\\r\\n<!-- </mat-checkbox>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div class=\\\"relations-list old-relations\\\">-->\\r\\n<!-- <div class=\\\"mat-body-1\\\" style=\\\"padding-bottom: 10px; color: rgba(0,0,0,0.57);\\\">Relations</div>-->\\r\\n<!-- <div class=\\\"body\\\" [fxShow]=\\\"oldRelations().length\\\">-->\\r\\n<!-- <div class=\\\"row\\\" fxLayout=\\\"row\\\" fxLayoutAlign=\\\"start center\\\" formArrayName=\\\"oldRelations\\\" -->\\r\\n<!-- *ngFor=\\\"let relation of oldRelations().controls; let i = index;\\\">-->\\r\\n<!-- <div [formGroupName]=\\\"i\\\" class=\\\"mat-elevation-z2\\\" fxFlex fxLayout=\\\"row\\\" style=\\\"padding: 5px 0 5px 5px;\\\">-->\\r\\n<!-- <div fxFlex fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field class=\\\"mat-block\\\" style=\\\"min-width: 100px;\\\">-->\\r\\n<!-- <mat-label>Direction</mat-label>-->\\r\\n<!-- <mat-select formControlName=\\\"direction\\\" name=\\\"direction\\\">-->\\r\\n<!-- <mat-option *ngFor=\\\"let direction of entitySearchDirection | keyvalue\\\" [value]=\\\"direction.value\\\">-->\\r\\n<!-- {{ (\\\"relation.search-direction.\\\" + direction.value) | translate}}-->\\r\\n<!-- </mat-option>-->\\r\\n<!-- </mat-select>-->\\r\\n<!-- <mat-error *ngIf=\\\"relation.get('direction').hasError('required')\\\">-->\\r\\n<!-- Relation direction is required.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <tb-relation-type-autocomplete-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"relationType\\\"-->\\r\\n<!-- required=\\\"true\\\">-->\\r\\n<!-- </tb-relation-type-autocomplete>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayout.xs=\\\"column\\\">-->\\r\\n<!-- <tb-entity-select-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- required=\\\"true\\\"-->\\r\\n<!-- formControlName=\\\"relatedEntity\\\">-->\\r\\n<!-- </tb-entity-select>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"column\\\" fxLayoutAlign=\\\"center center\\\">-->\\r\\n<!-- <button mat-icon-button color=\\\"primary\\\"-->\\r\\n<!-- aria-label=\\\"Remove\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- (click)=\\\"removeOldRelation(i)\\\"-->\\r\\n<!-- matTooltip=\\\"Remove relation\\\"-->\\r\\n<!-- matTooltipPosition=\\\"above\\\">-->\\r\\n<!-- <mat-icon>close</mat-icon>-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div class=\\\"relations-list\\\">-->\\r\\n<!-- <div class=\\\"mat-body-1\\\" style=\\\"padding-bottom: 10px; color: rgba(0,0,0,0.57);\\\">New Relations</div>-->\\r\\n<!-- <div class=\\\"body\\\" [fxShow]=\\\"relations().length\\\">-->\\r\\n<!-- <div class=\\\"row\\\" fxLayout=\\\"row\\\" fxLayoutAlign=\\\"start center\\\" formArrayName=\\\"relations\\\" *ngFor=\\\"let relation of relations().controls; let i = index;\\\">-->\\r\\n<!-- <div [formGroupName]=\\\"i\\\" class=\\\"mat-elevation-z2\\\" fxFlex fxLayout=\\\"row\\\" style=\\\"padding: 5px 0 5px 5px;\\\">-->\\r\\n<!-- <div fxFlex fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field class=\\\"mat-block\\\" style=\\\"min-width: 100px;\\\">-->\\r\\n<!-- <mat-label>Direction</mat-label>-->\\r\\n<!-- <mat-select formControlName=\\\"direction\\\" name=\\\"direction\\\">-->\\r\\n<!-- <mat-option *ngFor=\\\"let direction of entitySearchDirection | keyvalue\\\" [value]=\\\"direction.value\\\">-->\\r\\n<!-- {{ (\\\"relation.search-direction.\\\" + direction.value) | translate}}-->\\r\\n<!-- </mat-option>-->\\r\\n<!-- </mat-select>-->\\r\\n<!-- <mat-error *ngIf=\\\"relation.get('direction').hasError('required')\\\">-->\\r\\n<!-- Relation direction is required.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <tb-relation-type-autocomplete-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"relationType\\\"-->\\r\\n<!-- [required]=\\\"true\\\">-->\\r\\n<!-- </tb-relation-type-autocomplete>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayout.xs=\\\"column\\\">-->\\r\\n<!-- <tb-entity-select-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- [required]=\\\"true\\\"-->\\r\\n<!-- formControlName=\\\"relatedEntity\\\">-->\\r\\n<!-- </tb-entity-select>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"column\\\" fxLayoutAlign=\\\"center center\\\">-->\\r\\n<!-- <button mat-icon-button color=\\\"primary\\\"-->\\r\\n<!-- aria-label=\\\"Remove\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- (click)=\\\"removeRelation(i)\\\"-->\\r\\n<!-- matTooltip=\\\"Remove relation\\\"-->\\r\\n<!-- matTooltipPosition=\\\"above\\\">-->\\r\\n<!-- <mat-icon>close</mat-icon>-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div>-->\\r\\n<!-- <button mat-raised-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- (click)=\\\"addRelation()\\\"-->\\r\\n<!-- matTooltip=\\\"Add Relation\\\"-->\\r\\n<!-- matTooltipPosition=\\\"above\\\">-->\\r\\n<!-- Add-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div mat-dialog-actions fxLayout=\\\"row\\\" fxLayoutAlign=\\\"end center\\\">-->\\r\\n<!-- <button mat-button mat-raised-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"submit\\\"-->\\r\\n<!-- [disabled]=\\\"(isLoading$ | async) || editEntityForm.invalid || !editEntityForm.dirty\\\">-->\\r\\n<!-- Save-->\\r\\n<!-- </button>-->\\r\\n<!-- <button mat-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- [disabled]=\\\"(isLoading$ | async)\\\"-->\\r\\n<!-- (click)=\\\"cancel()\\\" cdkFocusInitial>-->\\r\\n<!-- Cancel-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!--</form>-->\\r\\n<!---->\\r\\n<!--========================================================================-->\\r\\n<!--========================= Add entity example =========================-->\\r\\n<!--========================================================================-->\\r\\n<!---->\\r\\n<!--<form #addEntityForm=\\\"ngForm\\\" [formGroup]=\\\"addEntityFormGroup\\\"-->\\r\\n<!-- (ngSubmit)=\\\"save()\\\" class=\\\"add-entity-form\\\">-->\\r\\n<!-- <mat-toolbar fxLayout=\\\"row\\\" color=\\\"primary\\\">-->\\r\\n<!-- <h2>Add entity</h2>-->\\r\\n<!-- <span fxFlex></span>-->\\r\\n<!-- <button mat-icon-button (click)=\\\"cancel()\\\" type=\\\"button\\\">-->\\r\\n<!-- <mat-icon class=\\\"material-icons\\\">close</mat-icon>-->\\r\\n<!-- </button>-->\\r\\n<!-- </mat-toolbar>-->\\r\\n<!-- <mat-progress-bar color=\\\"warn\\\" mode=\\\"indeterminate\\\" *ngIf=\\\"isLoading$ | async\\\">-->\\r\\n<!-- </mat-progress-bar>-->\\r\\n<!-- <div style=\\\"height: 4px;\\\" *ngIf=\\\"!(isLoading$ | async)\\\"></div>-->\\r\\n<!-- <div mat-dialog-content fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Entity Name</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"entityName\\\" required>-->\\r\\n<!-- <mat-error *ngIf=\\\"addEntityFormGroup.get('entityName').hasError('required')\\\">-->\\r\\n<!-- Entity name is required.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Entity Label</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"entityLabel\\\" >-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <tb-entity-type-select-->\\r\\n<!-- class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"entityType\\\"-->\\r\\n<!-- [showLabel]=\\\"true\\\"-->\\r\\n<!-- [allowedEntityTypes]=\\\"allowedEntityTypes\\\"-->\\r\\n<!-- ></tb-entity-type-select>-->\\r\\n<!-- <tb-entity-subtype-autocomplete-->\\r\\n<!-- fxFlex *ngIf=\\\"addEntityFormGroup.get('entityType').value == 'ASSET'\\\"-->\\r\\n<!-- class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"type\\\"-->\\r\\n<!-- [required]=\\\"true\\\"-->\\r\\n<!-- [entityType]=\\\"'ASSET'\\\"-->\\r\\n<!-- ></tb-entity-subtype-autocomplete>-->\\r\\n<!-- <tb-entity-subtype-autocomplete-->\\r\\n<!-- fxFlex *ngIf=\\\"addEntityFormGroup.get('entityType').value != 'ASSET'\\\"-->\\r\\n<!-- class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"type\\\"-->\\r\\n<!-- [required]=\\\"true\\\"-->\\r\\n<!-- [entityType]=\\\"'DEVICE'\\\"-->\\r\\n<!-- ></tb-entity-subtype-autocomplete>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div formGroupName=\\\"attributes\\\" fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Latitude</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"latitude\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Longitude</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"longitude\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Address</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"address\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Owner</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"owner\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Integer Value</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"1\\\" matInput formControlName=\\\"number\\\">-->\\r\\n<!-- <mat-error *ngIf=\\\"addEntityFormGroup.get('attributes.number').hasError('pattern')\\\">-->\\r\\n<!-- Invalid integer value.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <div class=\\\"boolean-value-input\\\" fxLayout=\\\"column\\\" fxLayoutAlign=\\\"center start\\\" fxFlex>-->\\r\\n<!-- <label class=\\\"checkbox-label\\\">Boolean Value</label>-->\\r\\n<!-- <mat-checkbox formControlName=\\\"booleanValue\\\" style=\\\"margin-bottom: 40px;\\\">-->\\r\\n<!-- {{ (addEntityFormGroup.get('attributes.booleanValue').value ? \\\"value.true\\\" : \\\"value.false\\\") | translate }}-->\\r\\n<!-- </mat-checkbox>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div class=\\\"relations-list\\\">-->\\r\\n<!-- <div class=\\\"mat-body-1\\\" style=\\\"padding-bottom: 10px; color: rgba(0,0,0,0.57);\\\">Relations</div>-->\\r\\n<!-- <div class=\\\"body\\\" [fxShow]=\\\"relations().length\\\">-->\\r\\n<!-- <div class=\\\"row\\\" fxLayout=\\\"row\\\" fxLayoutAlign=\\\"start center\\\" formArrayName=\\\"relations\\\" *ngFor=\\\"let relation of relations().controls; let i = index;\\\">-->\\r\\n<!-- <div [formGroupName]=\\\"i\\\" class=\\\"mat-elevation-z2\\\" fxFlex fxLayout=\\\"row\\\" style=\\\"padding: 5px 0 5px 5px;\\\">-->\\r\\n<!-- <div fxFlex fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field class=\\\"mat-block\\\" style=\\\"min-width: 100px;\\\">-->\\r\\n<!-- <mat-label>Direction</mat-label>-->\\r\\n<!-- <mat-select formControlName=\\\"direction\\\" name=\\\"direction\\\">-->\\r\\n<!-- <mat-option *ngFor=\\\"let direction of entitySearchDirection | keyvalue\\\" [value]=\\\"direction.value\\\">-->\\r\\n<!-- {{ (\\\"relation.search-direction.\\\" + direction.value) | translate}}-->\\r\\n<!-- </mat-option>-->\\r\\n<!-- </mat-select>-->\\r\\n<!-- <mat-error *ngIf=\\\"relation.get('direction').hasError('required')\\\">-->\\r\\n<!-- Relation direction is required.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <tb-relation-type-autocomplete-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"relationType\\\"-->\\r\\n<!-- [required]=\\\"true\\\">-->\\r\\n<!-- </tb-relation-type-autocomplete>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayout.xs=\\\"column\\\">-->\\r\\n<!-- <tb-entity-select-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- [required]=\\\"true\\\"-->\\r\\n<!-- formControlName=\\\"relatedEntity\\\">-->\\r\\n<!-- </tb-entity-select>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"column\\\" fxLayoutAlign=\\\"center center\\\">-->\\r\\n<!-- <button mat-icon-button color=\\\"primary\\\"-->\\r\\n<!-- aria-label=\\\"Remove\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- (click)=\\\"removeRelation(i)\\\"-->\\r\\n<!-- matTooltip=\\\"Remove relation\\\"-->\\r\\n<!-- matTooltipPosition=\\\"above\\\">-->\\r\\n<!-- <mat-icon>close</mat-icon>-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div>-->\\r\\n<!-- <button mat-raised-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- (click)=\\\"addRelation()\\\"-->\\r\\n<!-- matTooltip=\\\"Add Relation\\\"-->\\r\\n<!-- matTooltipPosition=\\\"above\\\">-->\\r\\n<!-- Add-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div mat-dialog-actions fxLayout=\\\"row\\\" fxLayoutAlign=\\\"end center\\\">-->\\r\\n<!-- <button mat-button mat-raised-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"submit\\\"-->\\r\\n<!-- [disabled]=\\\"(isLoading$ | async) || addEntityForm.invalid || !addEntityForm.dirty\\\">-->\\r\\n<!-- Create-->\\r\\n<!-- </button>-->\\r\\n<!-- <button mat-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- [disabled]=\\\"(isLoading$ | async)\\\"-->\\r\\n<!-- (click)=\\\"cancel()\\\" cdkFocusInitial>-->\\r\\n<!-- Cancel-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!--</form>-->\\r\\n\",\"customCss\":\"/*=======================================================================*/\\r\\n/*========== There are two examples: for edit and add entity ==========*/\\r\\n/*=======================================================================*/\\r\\n/*======================== Edit entity example ========================*/\\r\\n/*=======================================================================*/\\r\\n/*\\r\\n.edit-entity-form .boolean-value-input {\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.edit-entity-form .boolean-value-input .checkbox-label {\\r\\n margin-bottom: 8px;\\r\\n color: rgba(0,0,0,0.54);\\r\\n font-size: 12px;\\r\\n}\\r\\n\\r\\n.relations-list .header {\\r\\n padding-right: 5px;\\r\\n padding-bottom: 5px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .header .cell {\\r\\n padding-right: 5px;\\r\\n padding-left: 5px;\\r\\n font-size: 12px;\\r\\n font-weight: 700;\\r\\n color: rgba(0, 0, 0, .54);\\r\\n white-space: nowrap;\\r\\n}\\r\\n\\r\\n.relations-list .mat-form-field-infix {\\r\\n width: auto !important;\\r\\n}\\r\\n\\r\\n.relations-list .body {\\r\\n padding-right: 5px;\\r\\n padding-bottom: 15px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .row {\\r\\n padding-top: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .cell {\\r\\n padding-right: 5px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .md-button {\\r\\n margin: 0;\\r\\n}\\r\\n*/\\r\\n/*========================================================================*/\\r\\n/*========================= Add entity example =========================*/\\r\\n/*========================================================================*/\\r\\n/*\\r\\n.add-entity-form .boolean-value-input {\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.add-entity-form .boolean-value-input .checkbox-label {\\r\\n margin-bottom: 8px;\\r\\n color: rgba(0,0,0,0.54);\\r\\n font-size: 12px;\\r\\n}\\r\\n\\r\\n.relations-list .header {\\r\\n padding-right: 5px;\\r\\n padding-bottom: 5px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .header .cell {\\r\\n padding-right: 5px;\\r\\n padding-left: 5px;\\r\\n font-size: 12px;\\r\\n font-weight: 700;\\r\\n color: rgba(0, 0, 0, .54);\\r\\n white-space: nowrap;\\r\\n}\\r\\n\\r\\n.relations-list .mat-form-field-infix {\\r\\n width: auto !important;\\r\\n}\\r\\n\\r\\n.relations-list .body {\\r\\n padding-right: 5px;\\r\\n padding-bottom: 15px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .row {\\r\\n padding-top: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .cell {\\r\\n padding-right: 5px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .md-button {\\r\\n margin: 0;\\r\\n}\\r\\n*/\\r\\n\",\"customFunction\":\"\\nvar entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id\\n });\\n\\nwidgetContext.map.savePolygonLocation(entityDatasource[0],\\n null\\n ).subscribe(function succes() {\\n widgetContext.updateAliases();\\n });\",\"customResources\":[],\"id\":\"46bf69cd-8906-234c-a879-e2e4c92f5b67\"}]},\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"displayTimewindow\":true}"
390 390 }
391 391 },
392 392 {
393   - "alias": "web_camera_input",
394   - "name": "Web Camera Input",
  393 + "alias": "markers_placement_openstreetmap",
  394 + "name": "Markers Placement - OpenStreetMap",
395 395 "descriptor": {
396 396 "type": "latest",
397   - "sizeX": 7.5,
398   - "sizeY": 3,
  397 + "sizeX": 8.5,
  398 + "sizeY": 6.5,
399 399 "resources": [],
400   - "templateHtml": "<tb-web-camera-widget \n [ctx]=\"ctx\">\n</tb-web-camera-widget>",
401   - "templateCss": "",
402   - "controllerScript": "self.onInit = function() {\n}\n\nself.onDataUpdated = function() {\n self.ctx.$scope.webCameraInputWidget.onDataUpdated();\n}\n\nself.typeParameters = function() {\n return {\n maxDatasources: 1,\n maxDataKeys: 1,\n singleEntity: true\n }\n}\n\nself.onDestroy = function() {\n}\n",
403   - "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Web Camera\",\n \"properties\": {\n \"widgetTitle\": {\n \"title\": \"Widget title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"imageFormat\": {\n \"title\": \"Image Format\",\n \"type\": \"string\",\n \"default\": \"image/png\"\n },\n \"imageQuality\":{\n \"title\":\"Image quality that use lossy compression such as jpeg and webp\",\n \"type\":\"number\",\n \"default\": 0.92,\n \"min\": 0,\n \"max\": 1\n },\n \"maxWidth\": {\n \"title\": \"The maximal image width\",\n \"type\": \"number\",\n \"default\": 640\n }, \n \"maxHeight\": {\n \"title\": \"The maximal image heigth\",\n \"type\": \"number\",\n \"default\": 480\n }\n },\n \"required\": []\n },\n \"form\": [\n \"widgetTitle\",\n {\n \"key\": \"imageFormat\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"image/jpeg\",\n \"label\": \"JPEG\"\n },\n {\n \"value\": \"image/png\",\n \"label\": \"PNG\"\n },\n {\n \"value\": \"image/webp\",\n \"label\": \"WEBP\"\n }\n ]\n },\n \"imageQuality\",\n \"maxWidth\",\n \"maxHeight\"\n ]\n}",
  400 + "templateHtml": "",
  401 + "templateCss": ".leaflet-zoom-box {\n\tz-index: 9;\n}\n\n.leaflet-pane { z-index: 4; }\n\n.leaflet-tile-pane { z-index: 2; }\n.leaflet-overlay-pane { z-index: 4; }\n.leaflet-shadow-pane { z-index: 5; }\n.leaflet-marker-pane { z-index: 6; }\n.leaflet-tooltip-pane { z-index: 7; }\n.leaflet-popup-pane { z-index: 8; }\n\n.leaflet-map-pane canvas { z-index: 1; }\n.leaflet-map-pane svg { z-index: 2; }\n\n.leaflet-control {\n\tz-index: 9;\n}\n.leaflet-top,\n.leaflet-bottom {\n\tz-index: 11;\n}\n\n.tb-marker-label {\n border: none;\n background: none;\n box-shadow: none;\n}\n\n.tb-marker-label:before {\n border: none;\n background: none;\n}\n",
  402 + "controllerScript": "self.onInit = function() {\n self.ctx.map = new TbMapWidgetV2('openstreet-map', false, self.ctx, null, true);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbMapWidgetV2.settingsSchema('openstreet-map');\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbMapWidgetV2.dataKeySettingsSchema('openstreet-map');\n}\n\nself.actionSources = function() {\n return TbMapWidgetV2.actionSources();\n}\n\nself.onDestroy = function() {\n}\n\nself.typeParameters = function() {\n return {\n hasDataPageLink: true\n };\n}",
  403 + "settingsSchema": "{}",
404 404 "dataKeySettingsSchema": "{}\n",
405   - "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Web Camera Input\",\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"dropShadow\":true,\"enableFullscreen\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"displayTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
  405 + "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.05427416942713381,\"funcBody\":\"var value = prevValue || 15.833293;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.680594833308841,\"funcBody\":\"var value = prevValue || -90.454350;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]},{\"type\":\"function\",\"name\":\"Second point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#607d8b\",\"settings\":{},\"_hash\":0.7867521952070078,\"funcBody\":\"var value = prevValue || 14.450463;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#9c27b0\",\"settings\":{},\"_hash\":0.7040053227577256,\"funcBody\":\"var value = prevValue || -84.845334;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"fitMapBounds\":true,\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"label\":\"${entityName}\",\"tooltipPattern\":\"<b>${entityName}</b><br/><br/><b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><br/><link-act name='delete'>Delete</link-act>\",\"markerImageSize\":34,\"useColorFunction\":false,\"markerImages\":[],\"useMarkerImageFunction\":false,\"color\":\"#fe7569\",\"mapProvider\":\"OpenStreetMap.Mapnik\",\"showTooltip\":true,\"autocloseTooltip\":true,\"defaultCenterPosition\":\"0,0\",\"customProviderTileUrl\":\"https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png\",\"showTooltipAction\":\"click\",\"polygonKeyName\":\"coordinates\",\"polygonOpacity\":0.5,\"polygonStrokeOpacity\":1,\"polygonStrokeWeight\":1,\"zoomOnClick\":true,\"showCoverageOnHover\":true,\"animate\":true,\"maxClusterRadius\":80,\"removeOutsideVisibleBounds\":true,\"defaultZoomLevel\":5,\"provider\":\"openstreet-map\",\"draggableMarker\":true,\"editablePolygon\":true,\"mapPageSize\":16384},\"title\":\"Markers Placement - OpenStreetMap\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{\"tooltipAction\":[{\"name\":\"delete\",\"icon\":\"more_horiz\",\"type\":\"custom\",\"customFunction\":\"\\nvar entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id\\n });\\n\\nwidgetContext.map.setMarkerLocation(entityDatasource[0],\\n null,\\n null\\n ).subscribe(function succes() {\\n widgetContext.updateAliases();\\n });\",\"id\":\"54c293c4-9ca6-e34f-dc6a-0271944c1c66\"},{\"name\":\"delete_polygon\",\"icon\":\"more_horiz\",\"type\":\"customPretty\",\"customHtml\":\"<!--=======================================================================-->\\r\\n<!--===== There are two example templates: for edit and add entity =====-->\\r\\n<!--=======================================================================-->\\r\\n<!--======================== Edit entity example ========================-->\\r\\n<!--=======================================================================-->\\r\\n<!-- -->\\r\\n<!--<form #editEntityForm=\\\"ngForm\\\" [formGroup]=\\\"editEntityFormGroup\\\"-->\\r\\n<!-- (ngSubmit)=\\\"save()\\\" class=\\\"edit-entity-form\\\">-->\\r\\n<!-- <mat-toolbar fxLayout=\\\"row\\\" color=\\\"primary\\\">-->\\r\\n<!-- <h2>Edit {{entityType.toLowerCase()}} {{entityName}}</h2>-->\\r\\n<!-- <span fxFlex></span>-->\\r\\n<!-- <button mat-icon-button (click)=\\\"cancel()\\\" type=\\\"button\\\">-->\\r\\n<!-- <mat-icon class=\\\"material-icons\\\">close</mat-icon>-->\\r\\n<!-- </button>-->\\r\\n<!-- </mat-toolbar>-->\\r\\n<!-- <mat-progress-bar color=\\\"warn\\\" mode=\\\"indeterminate\\\" *ngIf=\\\"isLoading$ | async\\\">-->\\r\\n<!-- </mat-progress-bar>-->\\r\\n<!-- <div style=\\\"height: 4px;\\\" *ngIf=\\\"!(isLoading$ | async)\\\"></div>-->\\r\\n<!-- <div mat-dialog-content fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Entity Name</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"entityName\\\" required readonly=\\\"\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Entity Label</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"entityLabel\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Entity Type</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"entityType\\\" readonly>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Type</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"type\\\" readonly>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div formGroupName=\\\"attributes\\\" fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Latitude</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"latitude\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Longitude</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"longitude\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Address</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"address\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Owner</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"owner\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Integer Value</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"1\\\" matInput formControlName=\\\"number\\\">-->\\r\\n<!-- <mat-error *ngIf=\\\"editEntityFormGroup.get('attributes.number').hasError('pattern')\\\">-->\\r\\n<!-- Invalid integer value.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <div class=\\\"boolean-value-input\\\" fxLayout=\\\"column\\\" fxLayoutAlign=\\\"center start\\\" fxFlex>-->\\r\\n<!-- <label class=\\\"checkbox-label\\\">Boolean Value</label>-->\\r\\n<!-- <mat-checkbox formControlName=\\\"booleanValue\\\" style=\\\"margin-bottom: 40px;\\\">-->\\r\\n<!-- {{ (editEntityFormGroup.get('attributes.booleanValue').value ? \\\"value.true\\\" : \\\"value.false\\\") | translate }}-->\\r\\n<!-- </mat-checkbox>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div class=\\\"relations-list old-relations\\\">-->\\r\\n<!-- <div class=\\\"mat-body-1\\\" style=\\\"padding-bottom: 10px; color: rgba(0,0,0,0.57);\\\">Relations</div>-->\\r\\n<!-- <div class=\\\"body\\\" [fxShow]=\\\"oldRelations().length\\\">-->\\r\\n<!-- <div class=\\\"row\\\" fxLayout=\\\"row\\\" fxLayoutAlign=\\\"start center\\\" formArrayName=\\\"oldRelations\\\" -->\\r\\n<!-- *ngFor=\\\"let relation of oldRelations().controls; let i = index;\\\">-->\\r\\n<!-- <div [formGroupName]=\\\"i\\\" class=\\\"mat-elevation-z2\\\" fxFlex fxLayout=\\\"row\\\" style=\\\"padding: 5px 0 5px 5px;\\\">-->\\r\\n<!-- <div fxFlex fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field class=\\\"mat-block\\\" style=\\\"min-width: 100px;\\\">-->\\r\\n<!-- <mat-label>Direction</mat-label>-->\\r\\n<!-- <mat-select formControlName=\\\"direction\\\" name=\\\"direction\\\">-->\\r\\n<!-- <mat-option *ngFor=\\\"let direction of entitySearchDirection | keyvalue\\\" [value]=\\\"direction.value\\\">-->\\r\\n<!-- {{ (\\\"relation.search-direction.\\\" + direction.value) | translate}}-->\\r\\n<!-- </mat-option>-->\\r\\n<!-- </mat-select>-->\\r\\n<!-- <mat-error *ngIf=\\\"relation.get('direction').hasError('required')\\\">-->\\r\\n<!-- Relation direction is required.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <tb-relation-type-autocomplete-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"relationType\\\"-->\\r\\n<!-- required=\\\"true\\\">-->\\r\\n<!-- </tb-relation-type-autocomplete>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayout.xs=\\\"column\\\">-->\\r\\n<!-- <tb-entity-select-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- required=\\\"true\\\"-->\\r\\n<!-- formControlName=\\\"relatedEntity\\\">-->\\r\\n<!-- </tb-entity-select>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"column\\\" fxLayoutAlign=\\\"center center\\\">-->\\r\\n<!-- <button mat-icon-button color=\\\"primary\\\"-->\\r\\n<!-- aria-label=\\\"Remove\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- (click)=\\\"removeOldRelation(i)\\\"-->\\r\\n<!-- matTooltip=\\\"Remove relation\\\"-->\\r\\n<!-- matTooltipPosition=\\\"above\\\">-->\\r\\n<!-- <mat-icon>close</mat-icon>-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div class=\\\"relations-list\\\">-->\\r\\n<!-- <div class=\\\"mat-body-1\\\" style=\\\"padding-bottom: 10px; color: rgba(0,0,0,0.57);\\\">New Relations</div>-->\\r\\n<!-- <div class=\\\"body\\\" [fxShow]=\\\"relations().length\\\">-->\\r\\n<!-- <div class=\\\"row\\\" fxLayout=\\\"row\\\" fxLayoutAlign=\\\"start center\\\" formArrayName=\\\"relations\\\" *ngFor=\\\"let relation of relations().controls; let i = index;\\\">-->\\r\\n<!-- <div [formGroupName]=\\\"i\\\" class=\\\"mat-elevation-z2\\\" fxFlex fxLayout=\\\"row\\\" style=\\\"padding: 5px 0 5px 5px;\\\">-->\\r\\n<!-- <div fxFlex fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field class=\\\"mat-block\\\" style=\\\"min-width: 100px;\\\">-->\\r\\n<!-- <mat-label>Direction</mat-label>-->\\r\\n<!-- <mat-select formControlName=\\\"direction\\\" name=\\\"direction\\\">-->\\r\\n<!-- <mat-option *ngFor=\\\"let direction of entitySearchDirection | keyvalue\\\" [value]=\\\"direction.value\\\">-->\\r\\n<!-- {{ (\\\"relation.search-direction.\\\" + direction.value) | translate}}-->\\r\\n<!-- </mat-option>-->\\r\\n<!-- </mat-select>-->\\r\\n<!-- <mat-error *ngIf=\\\"relation.get('direction').hasError('required')\\\">-->\\r\\n<!-- Relation direction is required.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <tb-relation-type-autocomplete-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"relationType\\\"-->\\r\\n<!-- [required]=\\\"true\\\">-->\\r\\n<!-- </tb-relation-type-autocomplete>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayout.xs=\\\"column\\\">-->\\r\\n<!-- <tb-entity-select-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- [required]=\\\"true\\\"-->\\r\\n<!-- formControlName=\\\"relatedEntity\\\">-->\\r\\n<!-- </tb-entity-select>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"column\\\" fxLayoutAlign=\\\"center center\\\">-->\\r\\n<!-- <button mat-icon-button color=\\\"primary\\\"-->\\r\\n<!-- aria-label=\\\"Remove\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- (click)=\\\"removeRelation(i)\\\"-->\\r\\n<!-- matTooltip=\\\"Remove relation\\\"-->\\r\\n<!-- matTooltipPosition=\\\"above\\\">-->\\r\\n<!-- <mat-icon>close</mat-icon>-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div>-->\\r\\n<!-- <button mat-raised-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- (click)=\\\"addRelation()\\\"-->\\r\\n<!-- matTooltip=\\\"Add Relation\\\"-->\\r\\n<!-- matTooltipPosition=\\\"above\\\">-->\\r\\n<!-- Add-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div mat-dialog-actions fxLayout=\\\"row\\\" fxLayoutAlign=\\\"end center\\\">-->\\r\\n<!-- <button mat-button mat-raised-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"submit\\\"-->\\r\\n<!-- [disabled]=\\\"(isLoading$ | async) || editEntityForm.invalid || !editEntityForm.dirty\\\">-->\\r\\n<!-- Save-->\\r\\n<!-- </button>-->\\r\\n<!-- <button mat-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- [disabled]=\\\"(isLoading$ | async)\\\"-->\\r\\n<!-- (click)=\\\"cancel()\\\" cdkFocusInitial>-->\\r\\n<!-- Cancel-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!--</form>-->\\r\\n<!---->\\r\\n<!--========================================================================-->\\r\\n<!--========================= Add entity example =========================-->\\r\\n<!--========================================================================-->\\r\\n<!---->\\r\\n<!--<form #addEntityForm=\\\"ngForm\\\" [formGroup]=\\\"addEntityFormGroup\\\"-->\\r\\n<!-- (ngSubmit)=\\\"save()\\\" class=\\\"add-entity-form\\\">-->\\r\\n<!-- <mat-toolbar fxLayout=\\\"row\\\" color=\\\"primary\\\">-->\\r\\n<!-- <h2>Add entity</h2>-->\\r\\n<!-- <span fxFlex></span>-->\\r\\n<!-- <button mat-icon-button (click)=\\\"cancel()\\\" type=\\\"button\\\">-->\\r\\n<!-- <mat-icon class=\\\"material-icons\\\">close</mat-icon>-->\\r\\n<!-- </button>-->\\r\\n<!-- </mat-toolbar>-->\\r\\n<!-- <mat-progress-bar color=\\\"warn\\\" mode=\\\"indeterminate\\\" *ngIf=\\\"isLoading$ | async\\\">-->\\r\\n<!-- </mat-progress-bar>-->\\r\\n<!-- <div style=\\\"height: 4px;\\\" *ngIf=\\\"!(isLoading$ | async)\\\"></div>-->\\r\\n<!-- <div mat-dialog-content fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Entity Name</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"entityName\\\" required>-->\\r\\n<!-- <mat-error *ngIf=\\\"addEntityFormGroup.get('entityName').hasError('required')\\\">-->\\r\\n<!-- Entity name is required.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Entity Label</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"entityLabel\\\" >-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <tb-entity-type-select-->\\r\\n<!-- class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"entityType\\\"-->\\r\\n<!-- [showLabel]=\\\"true\\\"-->\\r\\n<!-- [allowedEntityTypes]=\\\"allowedEntityTypes\\\"-->\\r\\n<!-- ></tb-entity-type-select>-->\\r\\n<!-- <tb-entity-subtype-autocomplete-->\\r\\n<!-- fxFlex *ngIf=\\\"addEntityFormGroup.get('entityType').value == 'ASSET'\\\"-->\\r\\n<!-- class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"type\\\"-->\\r\\n<!-- [required]=\\\"true\\\"-->\\r\\n<!-- [entityType]=\\\"'ASSET'\\\"-->\\r\\n<!-- ></tb-entity-subtype-autocomplete>-->\\r\\n<!-- <tb-entity-subtype-autocomplete-->\\r\\n<!-- fxFlex *ngIf=\\\"addEntityFormGroup.get('entityType').value != 'ASSET'\\\"-->\\r\\n<!-- class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"type\\\"-->\\r\\n<!-- [required]=\\\"true\\\"-->\\r\\n<!-- [entityType]=\\\"'DEVICE'\\\"-->\\r\\n<!-- ></tb-entity-subtype-autocomplete>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div formGroupName=\\\"attributes\\\" fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Latitude</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"latitude\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Longitude</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"longitude\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Address</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"address\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Owner</mat-label>-->\\r\\n<!-- <input matInput formControlName=\\\"owner\\\">-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field fxFlex class=\\\"mat-block\\\">-->\\r\\n<!-- <mat-label>Integer Value</mat-label>-->\\r\\n<!-- <input type=\\\"number\\\" step=\\\"1\\\" matInput formControlName=\\\"number\\\">-->\\r\\n<!-- <mat-error *ngIf=\\\"addEntityFormGroup.get('attributes.number').hasError('pattern')\\\">-->\\r\\n<!-- Invalid integer value.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <div class=\\\"boolean-value-input\\\" fxLayout=\\\"column\\\" fxLayoutAlign=\\\"center start\\\" fxFlex>-->\\r\\n<!-- <label class=\\\"checkbox-label\\\">Boolean Value</label>-->\\r\\n<!-- <mat-checkbox formControlName=\\\"booleanValue\\\" style=\\\"margin-bottom: 40px;\\\">-->\\r\\n<!-- {{ (addEntityFormGroup.get('attributes.booleanValue').value ? \\\"value.true\\\" : \\\"value.false\\\") | translate }}-->\\r\\n<!-- </mat-checkbox>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div class=\\\"relations-list\\\">-->\\r\\n<!-- <div class=\\\"mat-body-1\\\" style=\\\"padding-bottom: 10px; color: rgba(0,0,0,0.57);\\\">Relations</div>-->\\r\\n<!-- <div class=\\\"body\\\" [fxShow]=\\\"relations().length\\\">-->\\r\\n<!-- <div class=\\\"row\\\" fxLayout=\\\"row\\\" fxLayoutAlign=\\\"start center\\\" formArrayName=\\\"relations\\\" *ngFor=\\\"let relation of relations().controls; let i = index;\\\">-->\\r\\n<!-- <div [formGroupName]=\\\"i\\\" class=\\\"mat-elevation-z2\\\" fxFlex fxLayout=\\\"row\\\" style=\\\"padding: 5px 0 5px 5px;\\\">-->\\r\\n<!-- <div fxFlex fxLayout=\\\"column\\\">-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\" fxLayout.xs=\\\"column\\\" fxLayoutGap.xs=\\\"0\\\">-->\\r\\n<!-- <mat-form-field class=\\\"mat-block\\\" style=\\\"min-width: 100px;\\\">-->\\r\\n<!-- <mat-label>Direction</mat-label>-->\\r\\n<!-- <mat-select formControlName=\\\"direction\\\" name=\\\"direction\\\">-->\\r\\n<!-- <mat-option *ngFor=\\\"let direction of entitySearchDirection | keyvalue\\\" [value]=\\\"direction.value\\\">-->\\r\\n<!-- {{ (\\\"relation.search-direction.\\\" + direction.value) | translate}}-->\\r\\n<!-- </mat-option>-->\\r\\n<!-- </mat-select>-->\\r\\n<!-- <mat-error *ngIf=\\\"relation.get('direction').hasError('required')\\\">-->\\r\\n<!-- Relation direction is required.-->\\r\\n<!-- </mat-error>-->\\r\\n<!-- </mat-form-field>-->\\r\\n<!-- <tb-relation-type-autocomplete-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- formControlName=\\\"relationType\\\"-->\\r\\n<!-- [required]=\\\"true\\\">-->\\r\\n<!-- </tb-relation-type-autocomplete>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"row\\\" fxLayout.xs=\\\"column\\\">-->\\r\\n<!-- <tb-entity-select-->\\r\\n<!-- fxFlex class=\\\"mat-block\\\"-->\\r\\n<!-- [required]=\\\"true\\\"-->\\r\\n<!-- formControlName=\\\"relatedEntity\\\">-->\\r\\n<!-- </tb-entity-select>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div fxLayout=\\\"column\\\" fxLayoutAlign=\\\"center center\\\">-->\\r\\n<!-- <button mat-icon-button color=\\\"primary\\\"-->\\r\\n<!-- aria-label=\\\"Remove\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- (click)=\\\"removeRelation(i)\\\"-->\\r\\n<!-- matTooltip=\\\"Remove relation\\\"-->\\r\\n<!-- matTooltipPosition=\\\"above\\\">-->\\r\\n<!-- <mat-icon>close</mat-icon>-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div>-->\\r\\n<!-- <button mat-raised-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- (click)=\\\"addRelation()\\\"-->\\r\\n<!-- matTooltip=\\\"Add Relation\\\"-->\\r\\n<!-- matTooltipPosition=\\\"above\\\">-->\\r\\n<!-- Add-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- </div>-->\\r\\n<!-- <div mat-dialog-actions fxLayout=\\\"row\\\" fxLayoutAlign=\\\"end center\\\">-->\\r\\n<!-- <button mat-button mat-raised-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"submit\\\"-->\\r\\n<!-- [disabled]=\\\"(isLoading$ | async) || addEntityForm.invalid || !addEntityForm.dirty\\\">-->\\r\\n<!-- Create-->\\r\\n<!-- </button>-->\\r\\n<!-- <button mat-button color=\\\"primary\\\"-->\\r\\n<!-- type=\\\"button\\\"-->\\r\\n<!-- [disabled]=\\\"(isLoading$ | async)\\\"-->\\r\\n<!-- (click)=\\\"cancel()\\\" cdkFocusInitial>-->\\r\\n<!-- Cancel-->\\r\\n<!-- </button>-->\\r\\n<!-- </div>-->\\r\\n<!--</form>-->\\r\\n\",\"customCss\":\"/*=======================================================================*/\\r\\n/*========== There are two examples: for edit and add entity ==========*/\\r\\n/*=======================================================================*/\\r\\n/*======================== Edit entity example ========================*/\\r\\n/*=======================================================================*/\\r\\n/*\\r\\n.edit-entity-form .boolean-value-input {\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.edit-entity-form .boolean-value-input .checkbox-label {\\r\\n margin-bottom: 8px;\\r\\n color: rgba(0,0,0,0.54);\\r\\n font-size: 12px;\\r\\n}\\r\\n\\r\\n.relations-list .header {\\r\\n padding-right: 5px;\\r\\n padding-bottom: 5px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .header .cell {\\r\\n padding-right: 5px;\\r\\n padding-left: 5px;\\r\\n font-size: 12px;\\r\\n font-weight: 700;\\r\\n color: rgba(0, 0, 0, .54);\\r\\n white-space: nowrap;\\r\\n}\\r\\n\\r\\n.relations-list .mat-form-field-infix {\\r\\n width: auto !important;\\r\\n}\\r\\n\\r\\n.relations-list .body {\\r\\n padding-right: 5px;\\r\\n padding-bottom: 15px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .row {\\r\\n padding-top: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .cell {\\r\\n padding-right: 5px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .md-button {\\r\\n margin: 0;\\r\\n}\\r\\n*/\\r\\n/*========================================================================*/\\r\\n/*========================= Add entity example =========================*/\\r\\n/*========================================================================*/\\r\\n/*\\r\\n.add-entity-form .boolean-value-input {\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.add-entity-form .boolean-value-input .checkbox-label {\\r\\n margin-bottom: 8px;\\r\\n color: rgba(0,0,0,0.54);\\r\\n font-size: 12px;\\r\\n}\\r\\n\\r\\n.relations-list .header {\\r\\n padding-right: 5px;\\r\\n padding-bottom: 5px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .header .cell {\\r\\n padding-right: 5px;\\r\\n padding-left: 5px;\\r\\n font-size: 12px;\\r\\n font-weight: 700;\\r\\n color: rgba(0, 0, 0, .54);\\r\\n white-space: nowrap;\\r\\n}\\r\\n\\r\\n.relations-list .mat-form-field-infix {\\r\\n width: auto !important;\\r\\n}\\r\\n\\r\\n.relations-list .body {\\r\\n padding-right: 5px;\\r\\n padding-bottom: 15px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .row {\\r\\n padding-top: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .cell {\\r\\n padding-right: 5px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\n.relations-list .body .md-button {\\r\\n margin: 0;\\r\\n}\\r\\n*/\\r\\n\",\"customFunction\":\"\\nvar entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id\\n });\\n\\nwidgetContext.map.savePolygonLocation(entityDatasource[0],\\n null\\n ).subscribe(function succes() {\\n widgetContext.updateAliases();\\n });\",\"customResources\":[],\"id\":\"6beb7bed-dfd8-388d-b60c-82988ab52f06\"}]},\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"displayTimewindow\":true}"
406 406 }
407 407 }
408 408 ]
... ...
... ... @@ -17,7 +17,7 @@
17 17 import L, {
18 18 FeatureGroup,
19 19 Icon,
20   - LatLngBounds,
  20 + LatLngBounds, LatLngExpression,
21 21 LatLngTuple,
22 22 markerClusterGroup,
23 23 MarkerClusterGroup,
... ... @@ -362,7 +362,7 @@ export default abstract class LeafletMap {
362 362 return L.latLng(lat, lng) as L.LatLng;
363 363 }
364 364
365   - convertPositionPolygon(expression: Array<[number, number]> | Array<Array<[number, number]>>) {
  365 + convertPositionPolygon(expression:LatLngExpression[][] | LatLngExpression[][][]) {
366 366 return (expression as Array<any>).map((el) => {
367 367 if (el.length === 2 && !el.some(isNaN)) {
368 368 return el;
... ... @@ -395,7 +395,7 @@ export default abstract class LeafletMap {
395 395 }
396 396 }
397 397
398   - convertPolygonToCustomFormat(expression: Array<Array<any>>): object {
  398 + convertPolygonToCustomFormat(expression: any[][]): object {
399 399 return {
400 400 [this.options.polygonKeyName] : this.convertToPolygonFormat(expression)
401 401 }
... ...
... ... @@ -14,7 +14,7 @@
14 14 /// limitations under the License.
15 15 ///
16 16
17   -import L, { LatLngBounds, LatLngLiteral, LatLngTuple } from 'leaflet';
  17 +import L, {LatLngBounds, LatLngExpression, LatLngLiteral, LatLngTuple} from 'leaflet';
18 18 import LeafletMap from '../leaflet-map';
19 19 import { MapImage, PosFuncton, UnitedMapSettings } from '../map-models';
20 20 import { Observable, ReplaySubject } from 'rxjs';
... ... @@ -224,7 +224,7 @@ export class ImageMap extends LeafletMap {
224 224 expression.y * this.height);
225 225 }
226 226
227   - convertPositionPolygon(expression: Array<[number, number]> | Array<Array<[number, number]>>) {
  227 + convertPositionPolygon(expression: LatLngExpression[][] | LatLngExpression[][][]) {
228 228 return (expression as Array<any>).map((el) => {
229 229 if (el.length === 2 && !el.some(isNaN)) {
230 230 return this.pointToLatLng(
... ... @@ -269,7 +269,7 @@ export class ImageMap extends LeafletMap {
269 269 }
270 270 }
271 271
272   - convertPolygonToCustomFormat(expression: Array<Array<any>>): object {
  272 + convertPolygonToCustomFormat(expression: any[][]): object {
273 273 return {
274 274 [this.options.polygonKeyName] : this.convertToPolygonFormat(expression)
275 275 }
... ...