Commit 0f24e00fea6f0068e1d901a7b78a3964f77e2ae3

Authored by lsyer
2 parents 18a12c14 e6056ec8
... ... @@ -18,7 +18,7 @@
18 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 19 "settingsSchema": "{}",
20 20 "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,\"editablePolygon\":true,\"mapPageSize\":16384,\"showPolygon\":false,\"polygonTooltipPattern\":\"<b>${entityName}</b><br/><br/><b>TimeStamp:</b> ${coordinates|ts:7}<br/><br/><link-act name='delete'>Delete</link-act>\"},\"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\":\"var entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id;\\n });\\n\\nwidgetContext.map.setMarkerLocation(entityDatasource[0], null, null).subscribe();\",\"id\":\"54c293c4-9ca6-e34f-dc6a-0271944c1c66\"},{\"name\":\"delete_polygon\",\"icon\":\"more_horiz\",\"type\":\"custom\",\"customFunction\":\"var entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id\\n });\\n\\nwidgetContext.map.savePolygonLocation(entityDatasource[0], null).subscribe();\",\"id\":\"6beb7bed-dfd8-388d-b60c-82988ab52f06\"}]},\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"displayTimewindow\":true}"
  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,\"editablePolygon\":true,\"mapPageSize\":16384,\"showPolygon\":false,\"polygonTooltipPattern\":\"<b>${entityName}</b><br/><br/><b>TimeStamp:</b> ${coordinates|ts:7}<br/><br/><link-act name='delete'>Delete</link-act>\"},\"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\":\"var entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id;\\n });\\n\\nwidgetContext.map.setMarkerLocation(entityDatasource[0], null, null).subscribe(() => widgetContext.updateAliases());\",\"id\":\"54c293c4-9ca6-e34f-dc6a-0271944c1c66\"},{\"name\":\"delete_polygon\",\"icon\":\"more_horiz\",\"type\":\"custom\",\"customFunction\":\"var entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id\\n });\\n\\nwidgetContext.map.savePolygonLocation(entityDatasource[0], null).subscribe(() => widgetContext.updateAliases());\",\"id\":\"6beb7bed-dfd8-388d-b60c-82988ab52f06\"}]},\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"displayTimewindow\":true}"
22 22 }
23 23 },
24 24 {
... ... @@ -66,7 +66,7 @@
66 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 67 "settingsSchema": "{}",
68 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,\"editablePolygon\":true,\"mapPageSize\":16384,\"showPolygon\":false,\"polygonTooltipPattern\":\"<b>${entityName}</b><br/><br/><b>TimeStamp:</b> ${coordinates|ts:7}<br/><br/><link-act name='delete_polygon'>Delete</link-act>\"},\"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\":\"var entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id;\\n });\\n\\nwidgetContext.map.setMarkerLocation(entityDatasource[0], null, null).subscribe();\",\"id\":\"c39f512a-21c6-6b06-3aa1-715262c6553d\"},{\"name\":\"delete_polygon\",\"icon\":\"more_horiz\",\"type\":\"custom\",\"customFunction\":\"var entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id\\n });\\n\\nwidgetContext.map.savePolygonLocation(entityDatasource[0], null).subscribe();\",\"id\":\"94bf5ffd-b526-c6c3-ae3b-ab42191217d9\"}]},\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"displayTimewindow\":true}"
  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,\"editablePolygon\":true,\"mapPageSize\":16384,\"showPolygon\":false,\"polygonTooltipPattern\":\"<b>${entityName}</b><br/><br/><b>TimeStamp:</b> ${coordinates|ts:7}<br/><br/><link-act name='delete_polygon'>Delete</link-act>\"},\"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\":\"var entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id;\\n });\\n\\nwidgetContext.map.setMarkerLocation(entityDatasource[0], null, null).subscribe(() => widgetContext.updateAliases());\",\"id\":\"c39f512a-21c6-6b06-3aa1-715262c6553d\"},{\"name\":\"delete_polygon\",\"icon\":\"more_horiz\",\"type\":\"custom\",\"customFunction\":\"var entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id\\n });\\n\\nwidgetContext.map.savePolygonLocation(entityDatasource[0], null).subscribe(() => widgetContext.updateAliases());\",\"id\":\"94bf5ffd-b526-c6c3-ae3b-ab42191217d9\"}]},\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"displayTimewindow\":true}"
70 70 }
71 71 },
72 72 {
... ... @@ -370,7 +370,7 @@
370 370 "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 371 "settingsSchema": "{}",
372 372 "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,\"editablePolygon\":true,\"mapPageSize\":16384,\"showPolygon\":false,\"polygonTooltipPattern\":\"<b>${entityName}</b><br/><br/><b>TimeStamp:</b> ${coordinates|ts:7}<br/><br/><link-act name='delete_polygon'>Delete</link-act>\",\"showPolygonTooltip\":false},\"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\":\"var entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id;\\n });\\n\\nwidgetContext.map.setMarkerLocation(entityDatasource[0], null, null).subscribe();\",\"id\":\"8d3c0156-0a14-7a6f-0ddd-0ec16b9ffc91\"},{\"name\":\"delete_polygon\",\"icon\":\"more_horiz\",\"type\":\"custom\",\"customFunction\":\"var entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id\\n });\\n\\nwidgetContext.map.savePolygonLocation(entityDatasource[0], null).subscribe();\",\"id\":\"46bf69cd-8906-234c-a879-e2e4c92f5b67\"}]},\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"displayTimewindow\":true}"
  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,\"editablePolygon\":true,\"mapPageSize\":16384,\"showPolygon\":false,\"polygonTooltipPattern\":\"<b>${entityName}</b><br/><br/><b>TimeStamp:</b> ${coordinates|ts:7}<br/><br/><link-act name='delete_polygon'>Delete</link-act>\",\"showPolygonTooltip\":false},\"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\":\"var entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id;\\n });\\n\\nwidgetContext.map.setMarkerLocation(entityDatasource[0], null, null).subscribe(() => widgetContext.updateAliases());\",\"id\":\"8d3c0156-0a14-7a6f-0ddd-0ec16b9ffc91\"},{\"name\":\"delete_polygon\",\"icon\":\"more_horiz\",\"type\":\"custom\",\"customFunction\":\"var entityDatasource = widgetContext.map.map.datasources.filter(\\n function(entity) {\\n return entity.entityId === entityId.id\\n });\\n\\nwidgetContext.map.savePolygonLocation(entityDatasource[0], null).subscribe(() => widgetContext.updateAliases());\",\"id\":\"46bf69cd-8906-234c-a879-e2e4c92f5b67\"}]},\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"displayTimewindow\":true}"
374 374 }
375 375 },
376 376 {
... ...
... ... @@ -93,7 +93,6 @@ public class ThingsboardInstallService {
93 93 } else if ("3.0.1-cassandra".equals(upgradeFromVersion)) {
94 94 log.info("Migrating ThingsBoard latest timeseries data from cassandra to SQL database ...");
95 95 latestMigrateService.migrate();
96   - log.info("Updating system data...");
97 96 } else {
98 97 switch (upgradeFromVersion) {
99 98 case "1.2.3": //NOSONAR, Need to execute gradual upgrade starting from upgradeFromVersion
... ... @@ -182,13 +181,12 @@ public class ThingsboardInstallService {
182 181 }
183 182 databaseEntitiesUpgradeService.upgradeDatabase("3.1.1");
184 183 dataUpdateService.updateData("3.1.1");
185   - log.info("Updating system data...");
186   - systemDataLoaderService.updateSystemWidgets();
187 184 systemDataLoaderService.createOAuth2Templates();
188   - break;
189 185 case "3.2.0":
190 186 log.info("Upgrading ThingsBoard from version 3.2.0 to 3.2.1 ...");
191 187 databaseEntitiesUpgradeService.upgradeDatabase("3.2.0");
  188 + log.info("Updating system data...");
  189 + systemDataLoaderService.updateSystemWidgets();
192 190 break;
193 191 default:
194 192 throw new RuntimeException("Unable to upgrade ThingsBoard, unsupported fromVersion: " + upgradeFromVersion);
... ...
... ... @@ -17,7 +17,6 @@ package org.thingsboard.server.dao.audit;
17 17
18 18 import com.datastax.oss.driver.api.core.uuid.Uuids;
19 19 import com.fasterxml.jackson.databind.JsonNode;
20   -import com.fasterxml.jackson.databind.ObjectMapper;
21 20 import com.fasterxml.jackson.databind.node.ArrayNode;
22 21 import com.fasterxml.jackson.databind.node.ObjectNode;
23 22 import com.google.common.collect.Lists;
... ... @@ -50,6 +49,7 @@ import org.thingsboard.server.dao.device.provision.ProvisionRequest;
50 49 import org.thingsboard.server.dao.entity.EntityService;
51 50 import org.thingsboard.server.dao.exception.DataValidationException;
52 51 import org.thingsboard.server.dao.service.DataValidator;
  52 +import org.thingsboard.server.dao.util.mapping.JacksonUtil;
53 53
54 54 import java.io.PrintWriter;
55 55 import java.io.StringWriter;
... ... @@ -65,8 +65,6 @@ import static org.thingsboard.server.dao.service.Validator.validateId;
65 65 @ConditionalOnProperty(prefix = "audit-log", value = "enabled", havingValue = "true")
66 66 public class AuditLogServiceImpl implements AuditLogService {
67 67
68   - private static final ObjectMapper objectMapper = new ObjectMapper();
69   -
70 68 private static final String INCORRECT_TENANT_ID = "Incorrect tenantId ";
71 69 private static final int INSERTS_PER_ENTRY = 3;
72 70
... ... @@ -159,7 +157,7 @@ public class AuditLogServiceImpl implements AuditLogService {
159 157 private <E extends HasName, I extends EntityId> JsonNode constructActionData(I entityId, E entity,
160 158 ActionType actionType,
161 159 Object... additionalInfo) {
162   - ObjectNode actionData = objectMapper.createObjectNode();
  160 + ObjectNode actionData = JacksonUtil.newObjectNode();
163 161 switch (actionType) {
164 162 case ADDED:
165 163 case UPDATED:
... ... @@ -168,7 +166,7 @@ public class AuditLogServiceImpl implements AuditLogService {
168 166 case RELATIONS_DELETED:
169 167 case ASSIGNED_TO_TENANT:
170 168 if (entity != null) {
171   - ObjectNode entityNode = objectMapper.valueToTree(entity);
  169 + ObjectNode entityNode = (ObjectNode) JacksonUtil.valueToTree(entity);
172 170 if (entityId.getEntityType() == EntityType.DASHBOARD) {
173 171 entityNode.put("configuration", "");
174 172 }
... ... @@ -177,7 +175,7 @@ public class AuditLogServiceImpl implements AuditLogService {
177 175 if (entityId.getEntityType() == EntityType.RULE_CHAIN) {
178 176 RuleChainMetaData ruleChainMetaData = extractParameter(RuleChainMetaData.class, additionalInfo);
179 177 if (ruleChainMetaData != null) {
180   - ObjectNode ruleChainMetaDataNode = objectMapper.valueToTree(ruleChainMetaData);
  178 + ObjectNode ruleChainMetaDataNode = (ObjectNode) JacksonUtil.valueToTree(ruleChainMetaData);
181 179 actionData.set("metadata", ruleChainMetaDataNode);
182 180 }
183 181 }
... ... @@ -194,7 +192,7 @@ public class AuditLogServiceImpl implements AuditLogService {
194 192 String scope = extractParameter(String.class, 0, additionalInfo);
195 193 List<AttributeKvEntry> attributes = extractParameter(List.class, 1, additionalInfo);
196 194 actionData.put("scope", scope);
197   - ObjectNode attrsNode = objectMapper.createObjectNode();
  195 + ObjectNode attrsNode = JacksonUtil.newObjectNode();
198 196 if (attributes != null) {
199 197 for (AttributeKvEntry attr : attributes) {
200 198 attrsNode.put(attr.getKey(), attr.getValueAsString());
... ... @@ -225,7 +223,7 @@ public class AuditLogServiceImpl implements AuditLogService {
225 223 case CREDENTIALS_UPDATED:
226 224 actionData.put("entityId", entityId.toString());
227 225 DeviceCredentials deviceCredentials = extractParameter(DeviceCredentials.class, additionalInfo);
228   - actionData.set("credentials", objectMapper.valueToTree(deviceCredentials));
  226 + actionData.set("credentials", JacksonUtil.valueToTree(deviceCredentials));
229 227 break;
230 228 case ASSIGNED_TO_CUSTOMER:
231 229 strEntityId = extractParameter(String.class, 0, additionalInfo);
... ... @@ -246,7 +244,7 @@ public class AuditLogServiceImpl implements AuditLogService {
246 244 case RELATION_ADD_OR_UPDATE:
247 245 case RELATION_DELETED:
248 246 EntityRelation relation = extractParameter(EntityRelation.class, 0, additionalInfo);
249   - actionData.set("relation", objectMapper.valueToTree(relation));
  247 + actionData.set("relation", JacksonUtil.valueToTree(relation));
250 248 break;
251 249 case LOGIN:
252 250 case LOGOUT:
... ... @@ -264,7 +262,7 @@ public class AuditLogServiceImpl implements AuditLogService {
264 262 case PROVISION_FAILURE:
265 263 ProvisionRequest request = extractParameter(ProvisionRequest.class, additionalInfo);
266 264 if (request != null) {
267   - actionData.set("provisionRequest", objectMapper.valueToTree(request));
  265 + actionData.set("provisionRequest", JacksonUtil.valueToTree(request));
268 266 }
269 267 break;
270 268 case TIMESERIES_UPDATED:
... ... @@ -275,7 +273,7 @@ public class AuditLogServiceImpl implements AuditLogService {
275 273 updatedTimeseries.stream()
276 274 .collect(Collectors.groupingBy(TsKvEntry::getTs))
277 275 .forEach((k, v) -> {
278   - ObjectNode element = objectMapper.createObjectNode();
  276 + ObjectNode element = JacksonUtil.newObjectNode();
279 277 element.put("ts", k);
280 278 ObjectNode values = element.putObject("values");
281 279 v.forEach(kvEntry -> values.put(kvEntry.getKey(), kvEntry.getValueAsString()));
... ...
... ... @@ -66,7 +66,7 @@ public class JacksonUtil {
66 66 throw new IllegalArgumentException(e);
67 67 }
68 68 }
69   -
  69 +
70 70 public static ObjectNode newObjectNode(){
71 71 return OBJECT_MAPPER.createObjectNode();
72 72 }
... ...
... ... @@ -58,7 +58,7 @@ frontend http-in
58 58
59 59 acl transport_http_acl path_beg /api/v1/
60 60 acl letsencrypt_http_acl path_beg /.well-known/acme-challenge/
61   - acl tb_api_acl path_beg /api/ /swagger /webjars /v2/ /static/rulenode/ /oauth2/ /login/oauth2/
  61 + acl tb_api_acl path_beg /api/ /swagger /webjars /v2/ /static/rulenode/ /oauth2/ /login/oauth2/ /static/widgets/
62 62
63 63 redirect scheme https if !letsencrypt_http_acl !transport_http_acl { env(FORCE_HTTPS_REDIRECT) -m str true }
64 64
... ... @@ -76,7 +76,7 @@ frontend https_in
76 76 reqadd X-Forwarded-Proto:\ https
77 77
78 78 acl transport_http_acl path_beg /api/v1/
79   - acl tb_api_acl path_beg /api/ /swagger /webjars /v2/ /static/rulenode/ /oauth2/ /login/oauth2/
  79 + acl tb_api_acl path_beg /api/ /swagger /webjars /v2/ /static/rulenode/ /oauth2/ /login/oauth2/ /static/widgets/
80 80
81 81 use_backend tb-http-backend if transport_http_acl
82 82 use_backend tb-api-backend if tb_api_acl
... ...
... ... @@ -64,6 +64,7 @@ class ProfileState {
64 64 alarmSettings.clear();
65 65 alarmCreateKeys.clear();
66 66 alarmClearKeys.clear();
  67 + entityKeys.clear();
67 68 if (deviceProfile.getProfileData().getAlarms() != null) {
68 69 alarmSettings.addAll(deviceProfile.getProfileData().getAlarms());
69 70 for (DeviceProfileAlarm alarm : deviceProfile.getProfileData().getAlarms()) {
... ...
... ... @@ -16,7 +16,7 @@
16 16 #
17 17
18 18 usage() {
19   - echo "This script generates client public/private rey pair, extracts them to a no-password RSA pem file,"
  19 + echo "This script generates client public/private key pair, extracts them to a no-password pem file,"
20 20 echo "and imports server public key to client keystore"
21 21 echo "usage: ./client.keygen.sh [-p file]"
22 22 echo " -p | --props | --properties file Properties file. default value is ./keygen.properties"
... ... @@ -70,6 +70,20 @@ while :
70 70 done
71 71 fi
72 72
  73 +OPENSSL_CMD=""
  74 +case $CLIENT_KEY_ALG in
  75 +RSA)
  76 + OPENSSL_CMD="rsa"
  77 + ;;
  78 +EC)
  79 + OPENSSL_CMD="ec"
  80 + ;;
  81 +esac
  82 +if [ -z "$OPENSSL_CMD" ]; then
  83 + echo "Unexpected CLIENT_KEY_ALG. Exiting."
  84 + exit 0
  85 +fi
  86 +
73 87 echo "Generating SSL Key Pair..."
74 88
75 89 keytool -genkeypair -v \
... ... @@ -77,8 +91,8 @@ keytool -genkeypair -v \
77 91 -keystore $CLIENT_FILE_PREFIX.jks \
78 92 -keypass $CLIENT_KEY_PASSWORD \
79 93 -storepass $CLIENT_KEYSTORE_PASSWORD \
80   - -keyalg RSA \
81   - -keysize 2048 \
  94 + -keyalg $CLIENT_KEY_ALG \
  95 + -keysize $CLIENT_KEY_SIZE\
82 96 -validity 9999 \
83 97 -dname "CN=$DOMAIN_SUFFIX, OU=$ORGANIZATIONAL_UNIT, O=$ORGANIZATION, L=$CITY, ST=$STATE_OR_PROVINCE, C=$TWO_LETTER_COUNTRY_CODE"
84 98
... ... @@ -110,7 +124,7 @@ keytool --importcert \
110 124 -noprompt
111 125
112 126 echo "Exporting no-password pem certificate"
113   -openssl rsa -in $CLIENT_FILE_PREFIX.pem -out $CLIENT_FILE_PREFIX.nopass.pem -passin pass:$CLIENT_KEY_PASSWORD
  127 +openssl $OPENSSL_CMD -in $CLIENT_FILE_PREFIX.pem -out $CLIENT_FILE_PREFIX.nopass.pem -passin pass:$CLIENT_KEY_PASSWORD
114 128 tail -n +$(($(grep -m1 -n -e '-----BEGIN CERTIFICATE' $CLIENT_FILE_PREFIX.pem | cut -d: -f1) )) \
115 129 $CLIENT_FILE_PREFIX.pem >> $CLIENT_FILE_PREFIX.nopass.pem
116 130
... ...
... ... @@ -26,6 +26,8 @@ SERVER_KEY_PASSWORD=server_key_password
26 26
27 27 SERVER_KEY_ALIAS="serveralias"
28 28 SERVER_FILE_PREFIX="mqttserver"
  29 +SERVER_KEY_ALG="RSA"
  30 +SERVER_KEY_SIZE="2048"
29 31 SERVER_KEYSTORE_DIR="/etc/thingsboard/conf"
30 32
31 33 CLIENT_KEYSTORE_PASSWORD=password
... ... @@ -33,4 +35,5 @@ CLIENT_KEY_PASSWORD=password
33 35
34 36 CLIENT_KEY_ALIAS="clientalias"
35 37 CLIENT_FILE_PREFIX="mqttclient"
36   -
  38 +CLIENT_KEY_ALG="RSA"
  39 +CLIENT_KEY_SIZE="2048"
... ...
... ... @@ -92,8 +92,8 @@ keytool -genkeypair -v \
92 92 -keystore $SERVER_FILE_PREFIX.jks \
93 93 -keypass $SERVER_KEY_PASSWORD \
94 94 -storepass $SERVER_KEYSTORE_PASSWORD \
95   - -keyalg RSA \
96   - -keysize 2048 \
  95 + -keyalg $SERVER_KEY_ALG \
  96 + -keysize $SERVER_KEY_SIZE \
97 97 -validity 9999
98 98
99 99 status=$?
... ...
... ... @@ -26,15 +26,15 @@ const PROXY_CONFIG = {
26 26 "target": ruleNodeUiforwardUrl,
27 27 "secure": false,
28 28 },
29   - "/oauth2": {
  29 + "/static/widgets": {
30 30 "target": forwardUrl,
31 31 "secure": false,
32 32 },
33   - "/login/oauth2": {
  33 + "/oauth2": {
34 34 "target": forwardUrl,
35 35 "secure": false,
36 36 },
37   - "/static": {
  37 + "/login/oauth2": {
38 38 "target": forwardUrl,
39 39 "secure": false,
40 40 },
... ...
... ... @@ -112,6 +112,12 @@
112 112 {{ 'widget-action.open-right-layout' | translate }}
113 113 </mat-checkbox>
114 114 </ng-template>
  115 + <ng-template [ngSwitchCase]="widgetActionFormGroup.get('type').value === widgetActionType.openDashboard ?
  116 + widgetActionFormGroup.get('type').value : ''">
  117 + <mat-checkbox formControlName="openNewBrowserTab">
  118 + {{ 'widget-action.open-new-browser-tab' | translate }}
  119 + </mat-checkbox>
  120 + </ng-template>
115 121 <ng-template [ngSwitchCase]="widgetActionFormGroup.get('type').value === widgetActionType.openDashboardState ||
116 122 widgetActionFormGroup.get('type').value === widgetActionType.updateDashboardState ||
117 123 widgetActionFormGroup.get('type').value === widgetActionType.openDashboard ?
... ...
... ... @@ -145,6 +145,10 @@ export class WidgetActionDialogComponent extends DialogComponent<WidgetActionDia
145 145 );
146 146 if (type === WidgetActionType.openDashboard) {
147 147 this.actionTypeFormGroup.addControl(
  148 + 'openNewBrowserTab',
  149 + this.fb.control(action ? action.openNewBrowserTab : false, [])
  150 + );
  151 + this.actionTypeFormGroup.addControl(
148 152 'targetDashboardId',
149 153 this.fb.control(action ? action.targetDashboardId : null,
150 154 [Validators.required])
... ...
... ... @@ -30,6 +30,7 @@ import { Datasource, DatasourceData } from '@shared/models/widget.models';
30 30 import _ from 'lodash';
31 31 import { mapProviderSchema, providerSets } from '@home/components/widget/lib/maps/schemes';
32 32 import { addCondition, mergeSchemes } from '@core/schema-utils';
  33 +import L, {Projection} from "leaflet";
33 34
34 35 export function getProviderSchema(mapProvider: MapProviders, ignoreImageMap = false) {
35 36 const providerSchema = _.cloneDeep(mapProviderSchema);
... ... @@ -443,3 +444,21 @@ export function createLoadingDiv(loadingText: string): JQuery<HTMLElement> {
443 444 </div>
444 445 `);
445 446 }
  447 +
  448 +export function checkLngLat(point: L.LatLng, southWest: L.LatLng, northEast: L.LatLng, offset = 0): L.LatLng {
  449 + const maxLngMap = northEast.lng - offset;
  450 + const minLngMap = southWest.lng + offset;
  451 + const maxLatMap = northEast.lat - offset;
  452 + const minLatMap = southWest.lat + offset;
  453 + if (point.lng > maxLngMap) {
  454 + point.lng = maxLngMap;
  455 + } else if (point.lng < minLngMap) {
  456 + point.lng = minLngMap;
  457 + }
  458 + if (point.lat > maxLatMap) {
  459 + point.lat = maxLatMap;
  460 + } else if (point.lat < minLatMap) {
  461 + point.lat = minLatMap;
  462 + }
  463 + return point;
  464 +}
... ...
... ... @@ -16,12 +16,12 @@
16 16
17 17 import L, {
18 18 FeatureGroup,
19   - Icon,
  19 + Icon, LatLng,
20 20 LatLngBounds,
21 21 LatLngTuple,
22 22 markerClusterGroup,
23 23 MarkerClusterGroup,
24   - MarkerClusterGroupOptions
  24 + MarkerClusterGroupOptions, Projection
25 25 } from 'leaflet';
26 26 import tinycolor from 'tinycolor2';
27 27 import 'leaflet-providers';
... ... @@ -46,6 +46,7 @@ import {
46 46 createTooltip,
47 47 } from '@home/components/widget/lib/maps/maps-utils';
48 48 import {
  49 + checkLngLat,
49 50 createLoadingDiv,
50 51 parseArray,
51 52 parseData,
... ... @@ -79,6 +80,8 @@ export default abstract class LeafletMap {
79 80 updatePending = false;
80 81 addMarkers: L.Marker[] = [];
81 82 addPolygons: L.Polygon[] = [];
  83 + southWest = new L.LatLng(-Projection.SphericalMercator['MAX_LATITUDE'], -180);
  84 + northEast = new L.LatLng(Projection.SphericalMercator['MAX_LATITUDE'], 180);
82 85
83 86 protected constructor(public ctx: WidgetContext,
84 87 public $container: HTMLElement,
... ... @@ -206,21 +209,30 @@ export default abstract class LeafletMap {
206 209
207 210 addPolygonControl() {
208 211 if (this.options.showPolygon && this.options.editablePolygon) {
209   - let mousePositionOnMap: L.LatLng[];
  212 + let polygonPoints: L.LatLng[];
210 213 let addPolygon: L.Control;
  214 + let mousePositionOnMap: LatLng;
211 215 this.map.on('mousemove', (e: L.LeafletMouseEvent) => {
212   - const polygonOffset = this.options.provider === MapProviders.image ? 10 : 0.01;
213   - const latlng1 = e.latlng;
214   - const latlng2 = L.latLng(e.latlng.lat, e.latlng.lng + polygonOffset);
215   - const latlng3 = L.latLng(e.latlng.lat - polygonOffset, e.latlng.lng);
216   - mousePositionOnMap = [latlng1, latlng2, latlng3];
  216 + mousePositionOnMap = e.latlng;
217 217 });
  218 +
218 219 const dragListener = (e: L.DragEndEvent) => {
219   - if (e.type === 'dragend' && mousePositionOnMap) {
220   - const newPolygon = L.polygon(mousePositionOnMap).addTo(this.map);
  220 + if (e.type === 'dragend') {
  221 + const polygonOffset = this.options.provider === MapProviders.image ? 10 : 0.01;
  222 +
  223 + let convert = this.convertToCustomFormat(mousePositionOnMap,polygonOffset);
  224 + mousePositionOnMap.lat = convert[this.options.latKeyName];
  225 + mousePositionOnMap.lng = convert[this.options.lngKeyName];
  226 +
  227 + const latlng1 = mousePositionOnMap;
  228 + const latlng2 = L.latLng(mousePositionOnMap.lat, mousePositionOnMap.lng + polygonOffset);
  229 + const latlng3 = L.latLng(mousePositionOnMap.lat - polygonOffset, mousePositionOnMap.lng);
  230 + polygonPoints = [latlng1, latlng2, latlng3];
  231 +
  232 + const newPolygon = L.polygon(polygonPoints).addTo(this.map);
221 233 this.addPolygons.push(newPolygon);
222 234 const datasourcesList = document.createElement('div');
223   - const customLatLng = {[this.options.polygonKeyName]: this.convertToPolygonFormat(mousePositionOnMap)};
  235 + const customLatLng = {[this.options.polygonKeyName]: this.convertToPolygonFormat(polygonPoints)};
224 236 const header = document.createElement('p');
225 237 header.appendChild(document.createTextNode('Select entity:'));
226 238 header.setAttribute('style', 'font-size: 14px; margin: 8px 0');
... ... @@ -414,12 +426,9 @@ export default abstract class LeafletMap {
414 426 }).filter(el => !!el);
415 427 }
416 428
417   - convertToCustomFormat(position: L.LatLng): object {
418   - if (position.lng > 180) {
419   - position.lng = 180;
420   - } else if (position.lng < -180) {
421   - position.lng = -180;
422   - }
  429 + convertToCustomFormat(position: L.LatLng, offset = 0): object {
  430 + position = checkLngLat(position, this.southWest, this.northEast, offset);
  431 +
423 432 return {
424 433 [this.options.latKeyName]: position.lat,
425 434 [this.options.lngKeyName]: position.lng
... ... @@ -728,6 +737,11 @@ export default abstract class LeafletMap {
728 737 if (e === undefined || (e.type !== 'editable:vertex:dragend' && e.type !== 'editable:vertex:deleted')) {
729 738 return;
730 739 }
  740 + if(this.options.provider !== MapProviders.image) {
  741 + for (let key in e.layer._latlngs[0]) {
  742 + e.layer._latlngs[0][key] = checkLngLat(e.layer._latlngs[0][key], this.southWest, this.northEast);
  743 + }
  744 + }
731 745 this.savePolygonLocation({ ...data, ...this.convertPolygonToCustomFormat(e.layer._latlngs) }).subscribe();
732 746 }
733 747
... ...
... ... @@ -19,7 +19,12 @@ import LeafletMap from '../leaflet-map';
19 19 import { MapImage, PosFuncton, UnitedMapSettings } from '../map-models';
20 20 import { Observable, ReplaySubject } from 'rxjs';
21 21 import { filter, map, mergeMap } from 'rxjs/operators';
22   -import { aspectCache, calculateNewPointCoordinate, parseFunction } from '@home/components/widget/lib/maps/common-maps-utils';
  22 +import {
  23 + aspectCache,
  24 + calculateNewPointCoordinate,
  25 + checkLngLat,
  26 + parseFunction
  27 +} from '@home/components/widget/lib/maps/common-maps-utils';
23 28 import { WidgetContext } from '@home/models/widget-component.models';
24 29 import { DataSet, DatasourceType, widgetType } from '@shared/models/widget.models';
25 30 import { DataKeyType } from '@shared/models/telemetry/telemetry.models';
... ... @@ -132,9 +137,9 @@ export class ImageMap extends LeafletMap {
132 137 updateBounds(updateImage?: boolean, lastCenterPos?) {
133 138 const w = this.width;
134 139 const h = this.height;
135   - let southWest = this.pointToLatLng(0, h);
136   - let northEast = this.pointToLatLng(w, 0);
137   - const bounds = new L.LatLngBounds(southWest, northEast);
  140 + this.southWest = this.pointToLatLng(0, h);
  141 + this.northEast = this.pointToLatLng(w, 0);
  142 + const bounds = new L.LatLngBounds(this.southWest, this.northEast);
138 143
139 144 if (updateImage && this.imageOverlay) {
140 145 this.imageOverlay.remove();
... ... @@ -147,8 +152,8 @@ export class ImageMap extends LeafletMap {
147 152 this.imageOverlay = L.imageOverlay(this.imageUrl, bounds).addTo(this.map);
148 153 }
149 154 const padding = 200 * maxZoom;
150   - southWest = this.pointToLatLng(-padding, h + padding);
151   - northEast = this.pointToLatLng(w + padding, -padding);
  155 + const southWest = this.pointToLatLng(-padding, h + padding);
  156 + const northEast = this.pointToLatLng(w + padding, -padding);
152 157 const maxBounds = new L.LatLngBounds(southWest, northEast);
153 158 this.map.setMaxBounds(maxBounds);
154 159 if (lastCenterPos) {
... ... @@ -187,7 +192,7 @@ export class ImageMap extends LeafletMap {
187 192 this.updateMarkers(this.markersData);
188 193 if (this.options.draggableMarker && this.addMarkers.length) {
189 194 this.addMarkers.forEach((marker) => {
190   - const prevPoint = this.convertToCustomFormat(marker.getLatLng(), prevWidth, prevHeight);
  195 + const prevPoint = this.convertToCustomFormat(marker.getLatLng(), null, prevWidth, prevHeight);
191 196 marker.setLatLng(this.convertPosition(prevPoint));
192 197 });
193 198 }
... ... @@ -257,11 +262,10 @@ export class ImageMap extends LeafletMap {
257 262 return L.CRS.Simple.latLngToPoint(latLng, maxZoom - 1);
258 263 }
259 264
260   - convertToCustomFormat(position: L.LatLng, width = this.width, height = this.height): object {
  265 + convertToCustomFormat(position: L.LatLng, offset = 0, width = this.width, height = this.height): object {
261 266 const point = this.latLngToPoint(position);
262 267 const customX = calculateNewPointCoordinate(point.x, width);
263 268 const customY = calculateNewPointCoordinate(point.y, height);
264   -
265 269 if (customX === 0) {
266 270 point.x = 0;
267 271 } else if (customX === 1) {
... ... @@ -273,7 +277,8 @@ export class ImageMap extends LeafletMap {
273 277 } else if (customY === 1) {
274 278 point.y = height;
275 279 }
276   - const customLatLng = this.pointToLatLng(point.x, point.y);
  280 +
  281 + const customLatLng = checkLngLat(this.pointToLatLng(point.x, point.y), this.southWest, this.northEast, offset);
277 282
278 283 return {
279 284 [this.options.xPosKeyName]: customX,
... ...
... ... @@ -1029,7 +1029,11 @@ export class WidgetComponent extends PageComponent implements OnInit, AfterViewI
1029 1029 } else {
1030 1030 url = `/dashboards/${targetDashboardId}?state=${state}`;
1031 1031 }
1032   - this.router.navigateByUrl(url);
  1032 + if (descriptor.openNewBrowserTab) {
  1033 + window.open(url, '_blank');
  1034 + } else {
  1035 + this.router.navigateByUrl(url);
  1036 + }
1033 1037 break;
1034 1038 case WidgetActionType.custom:
1035 1039 const customFunction = descriptor.customFunction;
... ...
... ... @@ -344,6 +344,7 @@ export interface WidgetActionDescriptor extends CustomActionDescriptor {
344 344 targetDashboardId?: string;
345 345 targetDashboardStateId?: string;
346 346 openRightLayout?: boolean;
  347 + openNewBrowserTab?: boolean;
347 348 setEntityId?: boolean;
348 349 stateEntityParamName?: string;
349 350 }
... ...
... ... @@ -2257,7 +2257,8 @@
2257 2257 "target-dashboard-state-required": "Target dashboard state is required",
2258 2258 "set-entity-from-widget": "Set entity from widget",
2259 2259 "target-dashboard": "Target dashboard",
2260   - "open-right-layout": "Open right dashboard layout (mobile view)"
  2260 + "open-right-layout": "Open right dashboard layout (mobile view)",
  2261 + "open-new-browser-tab": "Open in a new browser tab"
2261 2262 },
2262 2263 "widgets-bundle": {
2263 2264 "current": "Current bundle",
... ...