Commit a1f2ef40e57fc34bba575961da1e140454899e68

Authored by Igor Kulikov
Committed by GitHub
2 parents 7ef75524 a1fb61ff

Merge pull request #3280 from MrKartoshka/editable-polygons

Editable polygons
@@ -18,7 +18,7 @@ @@ -18,7 +18,7 @@
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}", 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": "{}", 19 "settingsSchema": "{}",
20 "dataKeySettingsSchema": "{}\n", 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},\"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}" 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},\"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\":\"custom\",\"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 });\",\"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,7 +66,7 @@
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}", 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": "{}", 67 "settingsSchema": "{}",
68 "dataKeySettingsSchema": "{}\n", 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}" 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},\"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\":\"custom\",\"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 });\",\"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,7 +370,7 @@
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}", 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 "settingsSchema": "{}", 371 "settingsSchema": "{}",
372 "dataKeySettingsSchema": "{}\n", 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},\"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}" 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},\"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\":\"custom\",\"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 });\",\"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 {
@@ -58,6 +58,7 @@ @@ -58,6 +58,7 @@
58 "jstree-bootstrap-theme": "^1.0.1", 58 "jstree-bootstrap-theme": "^1.0.1",
59 "jszip": "^3.4.0", 59 "jszip": "^3.4.0",
60 "leaflet": "^1.6.0", 60 "leaflet": "^1.6.0",
  61 + "leaflet-editable": "^1.2.0",
61 "leaflet-polylinedecorator": "^1.6.0", 62 "leaflet-polylinedecorator": "^1.6.0",
62 "leaflet-providers": "^1.10.1", 63 "leaflet-providers": "^1.10.1",
63 "leaflet.gridlayer.googlemutant": "0.10.0", 64 "leaflet.gridlayer.googlemutant": "0.10.0",
@@ -179,6 +179,74 @@ export default abstract class LeafletMap { @@ -179,6 +179,74 @@ export default abstract class LeafletMap {
179 } 179 }
180 } 180 }
181 181
  182 + addPolygonControl() {
  183 + if (this.options.editablePolygon) {
  184 + let mousePositionOnMap: L.LatLng[];
  185 + let addPolygon: L.Control;
  186 + this.map.on('mousemove', (e: L.LeafletMouseEvent) => {
  187 + const latlng1 = e.latlng;
  188 + const latlng2 = L.latLng(e.latlng.lat, e.latlng.lng + 10);
  189 + const latlng3 = L.latLng(e.latlng.lat-10, e.latlng.lng);
  190 + mousePositionOnMap = [latlng1,latlng2, latlng3 ];
  191 + });
  192 + const dragListener = (e: L.DragEndEvent) => {
  193 + if (e.type === 'dragend' && mousePositionOnMap) {
  194 + const icon = new L.Icon.Default();
  195 + icon.options.shadowSize = [0, 0];
  196 + const newPolygon = L.polygon(mousePositionOnMap).addTo(this.map);
  197 + const datasourcesList = document.createElement('div');
  198 + const customLatLng = {coordinates: this.convertToPolygonFormat(mousePositionOnMap)};
  199 + this.datasources.forEach(ds => {
  200 + const dsItem = document.createElement('p');
  201 + dsItem.appendChild(document.createTextNode(ds.entityName));
  202 + dsItem.setAttribute('style', 'font-size: 14px');
  203 + dsItem.onclick = () => {
  204 + const updatedEnttity = { ...ds, ...customLatLng };
  205 + this.savePolygonLocation(updatedEnttity).subscribe(() => {
  206 + this.map.removeLayer(newPolygon);
  207 + this.deletePolygon(ds.entityName);
  208 + });
  209 + }
  210 + datasourcesList.append(dsItem);
  211 + });
  212 + const deleteBtn = document.createElement('a');
  213 + deleteBtn.appendChild(document.createTextNode('Delete position'));
  214 + deleteBtn.setAttribute('color', 'red');
  215 + deleteBtn.onclick = () => {
  216 + this.map.removeLayer(newPolygon);
  217 + }
  218 + datasourcesList.append(deleteBtn);
  219 + const popup = L.popup();
  220 + popup.setContent(datasourcesList);
  221 + newPolygon.bindPopup(popup).openPopup();
  222 + }
  223 + addPolygon.setPosition('topright')
  224 + }
  225 + L.Control.AddPolygon = L.Control.extend({
  226 + onAdd() {
  227 + const img = L.DomUtil.create('img') as any;
  228 + img.src = `assets/add_polygon.svg`;
  229 + img.style.width = '32px';
  230 + img.style.height = '32px';
  231 + img.title = 'Drag and drop to add Polygon';
  232 + img.onclick = this.dragPolygonVertex;
  233 + img.draggable = true;
  234 + const draggableImg = new L.Draggable(img);
  235 + draggableImg.enable();
  236 + draggableImg.on('dragend', dragListener)
  237 + return img;
  238 + },
  239 + onRemove() {
  240 + },
  241 + dragPolygonVertex: this.dragPolygonVertex
  242 + } as any);
  243 + L.control.addPolygon = (opts) => {
  244 + return new L.Control.AddPolygon(opts);
  245 + }
  246 + addPolygon = L.control.addPolygon({ position: 'topright' }).addTo(this.map);
  247 + }
  248 + }
  249 +
182 public setLoading(loading: boolean) { 250 public setLoading(loading: boolean) {
183 if (this.loading !== loading) { 251 if (this.loading !== loading) {
184 this.loading = loading; 252 this.loading = loading;
@@ -205,6 +273,9 @@ export default abstract class LeafletMap { @@ -205,6 +273,9 @@ export default abstract class LeafletMap {
205 if (this.options.draggableMarker) { 273 if (this.options.draggableMarker) {
206 this.addMarkerControl(); 274 this.addMarkerControl();
207 } 275 }
  276 + if (this.options.editablePolygon) {
  277 + this.addPolygonControl();
  278 + }
208 if (this.options.useClusterMarkers) { 279 if (this.options.useClusterMarkers) {
209 this.map.addLayer(this.markersCluster); 280 this.map.addLayer(this.markersCluster);
210 } 281 }
@@ -218,6 +289,10 @@ export default abstract class LeafletMap { @@ -218,6 +289,10 @@ export default abstract class LeafletMap {
218 return of(null); 289 return of(null);
219 } 290 }
220 291
  292 + public savePolygonLocation(_e: FormattedData, coordinates?: Array<[number, number]>): Observable<any> {
  293 + return of(null);
  294 + }
  295 +
221 createLatLng(lat: number, lng: number): L.LatLng { 296 createLatLng(lat: number, lng: number): L.LatLng {
222 return L.latLng(lat, lng); 297 return L.latLng(lat, lng);
223 } 298 }
@@ -284,10 +359,16 @@ export default abstract class LeafletMap { @@ -284,10 +359,16 @@ export default abstract class LeafletMap {
284 return L.latLng(lat, lng) as L.LatLng; 359 return L.latLng(lat, lng) as L.LatLng;
285 } 360 }
286 361
287 - convertPositionPolygon(expression: Array<[number, number]>): L.LatLngExpression[] {  
288 - return expression.map((el) => {  
289 - return el.length === 2 && !el.some(isNaN) ? el : null  
290 - }).filter(el => !!el) 362 + convertPositionPolygon(expression: (LatLngTuple | LatLngTuple[] | LatLngTuple[][])[]) {
  363 + return (expression).map((el) => {
  364 + if (!Array.isArray(el[0]) && el.length === 2) {
  365 + return el;
  366 + } else if (Array.isArray(el) && el.length) {
  367 + return this.convertPositionPolygon(el as LatLngTuple[] | LatLngTuple[][]);
  368 + } else {
  369 + return null;
  370 + }
  371 + }).filter(el => !!el)
291 } 372 }
292 373
293 convertToCustomFormat(position: L.LatLng): object { 374 convertToCustomFormat(position: L.LatLng): object {
@@ -297,6 +378,26 @@ export default abstract class LeafletMap { @@ -297,6 +378,26 @@ export default abstract class LeafletMap {
297 } 378 }
298 } 379 }
299 380
  381 + convertToPolygonFormat(points: Array<any>): Array<any> {
  382 + if (points.length) {
  383 + return points.map(point=> {
  384 + if (point.length) {
  385 + return this.convertToPolygonFormat(point);
  386 + } else {
  387 + return [point.lat, point.lng];
  388 + }
  389 + })
  390 + } else {
  391 + return []
  392 + }
  393 + }
  394 +
  395 + convertPolygonToCustomFormat(expression: any[][]): object {
  396 + return {
  397 + [this.options.polygonKeyName] : this.convertToPolygonFormat(expression)
  398 + }
  399 + }
  400 +
300 updateData(drawRoutes: boolean, showPolygon: boolean) { 401 updateData(drawRoutes: boolean, showPolygon: boolean) {
301 this.drawRoutes = drawRoutes; 402 this.drawRoutes = drawRoutes;
302 this.showPolygon = showPolygon; 403 this.showPolygon = showPolygon;
@@ -311,7 +412,7 @@ export default abstract class LeafletMap { @@ -311,7 +412,7 @@ export default abstract class LeafletMap {
311 } 412 }
312 this.updateMarkers(formattedData, false); 413 this.updateMarkers(formattedData, false);
313 this.updateBoundsInternal(); 414 this.updateBoundsInternal();
314 - if (this.options.draggableMarker) { 415 + if (this.options.draggableMarker || this.options.editablePolygon) {
315 this.datasources = formattedData; 416 this.datasources = formattedData;
316 } 417 }
317 } else { 418 } else {
@@ -449,6 +550,16 @@ export default abstract class LeafletMap { @@ -449,6 +550,16 @@ export default abstract class LeafletMap {
449 return marker; 550 return marker;
450 } 551 }
451 552
  553 + deletePolygon(key: string) {
  554 + let polygon = this.polygons.get(key)?.leafletPoly;
  555 + if (polygon) {
  556 + this.map.removeLayer(polygon);
  557 + this.polygons.delete(key);
  558 + polygon = null;
  559 + }
  560 + return polygon;
  561 + }
  562 +
452 updatePoints(pointsData: FormattedData[], getTooltip: (point: FormattedData, setTooltip?: boolean) => string) { 563 updatePoints(pointsData: FormattedData[], getTooltip: (point: FormattedData, setTooltip?: boolean) => string) {
453 if (this.points) { 564 if (this.points) {
454 this.map.removeLayer(this.points); 565 this.map.removeLayer(this.points);
@@ -558,8 +669,13 @@ export default abstract class LeafletMap { @@ -558,8 +669,13 @@ export default abstract class LeafletMap {
558 }); 669 });
559 } 670 }
560 671
  672 + dragPolygonVertex = (e?, data = {} as FormattedData) => {
  673 + if (e === undefined || (e.type !== 'editable:vertex:dragend' && e.type !== 'editable:vertex:deleted')) return;
  674 + this.savePolygonLocation({ ...data, ...this.convertPolygonToCustomFormat(e.layer._latlngs) }).subscribe();
  675 + }
  676 +
561 createPolygon(polyData: FormattedData, dataSources: FormattedData[], settings: PolygonSettings, updateBounds = true) { 677 createPolygon(polyData: FormattedData, dataSources: FormattedData[], settings: PolygonSettings, updateBounds = true) {
562 - const polygon = new Polygon(this.map, polyData, dataSources, settings); 678 + const polygon = new Polygon(this.map, polyData, dataSources, settings, this.dragPolygonVertex);
563 if (updateBounds) { 679 if (updateBounds) {
564 const bounds = polygon.leafletPoly.getBounds(); 680 const bounds = polygon.leafletPoly.getBounds();
565 this.fitBounds(bounds); 681 this.fitBounds(bounds);
@@ -27,6 +27,7 @@ export type PosFuncton = (origXPos, origYPos) => { x, y }; @@ -27,6 +27,7 @@ export type PosFuncton = (origXPos, origYPos) => { x, y };
27 27
28 export type MapSettings = { 28 export type MapSettings = {
29 draggableMarker: boolean; 29 draggableMarker: boolean;
  30 + editablePolygon: boolean;
30 posFunction: PosFuncton; 31 posFunction: PosFuncton;
31 defaultZoomLevel?: number; 32 defaultZoomLevel?: number;
32 disableScrollZooming?: boolean; 33 disableScrollZooming?: boolean;
@@ -139,6 +140,7 @@ export type PolygonSettings = { @@ -139,6 +140,7 @@ export type PolygonSettings = {
139 usePolygonColorFunction: boolean; 140 usePolygonColorFunction: boolean;
140 polygonTooltipFunction: GenericFunction; 141 polygonTooltipFunction: GenericFunction;
141 polygonColorFunction?: GenericFunction; 142 polygonColorFunction?: GenericFunction;
  143 + editablePolygon: boolean;
142 } 144 }
143 145
144 export type PolylineSettings = { 146 export type PolylineSettings = {
@@ -233,6 +235,7 @@ export const defaultSettings: any = { @@ -233,6 +235,7 @@ export const defaultSettings: any = {
233 credentials: '', 235 credentials: '',
234 markerClusteringSetting: null, 236 markerClusteringSetting: null,
235 draggableMarker: false, 237 draggableMarker: false,
  238 + editablePolygon: false,
236 fitMapBounds: true, 239 fitMapBounds: true,
237 mapPageSize: DEFAULT_MAP_PAGE_SIZE 240 mapPageSize: DEFAULT_MAP_PAGE_SIZE
238 }; 241 };
@@ -78,6 +78,7 @@ export class MapWidgetController implements MapWidgetInterface { @@ -78,6 +78,7 @@ export class MapWidgetController implements MapWidgetInterface {
78 this.map = new MapClass(this.ctx, $element, this.settings); 78 this.map = new MapClass(this.ctx, $element, this.settings);
79 (this.ctx as any).mapInstance = this.map; 79 (this.ctx as any).mapInstance = this.map;
80 this.map.saveMarkerLocation = this.setMarkerLocation; 80 this.map.saveMarkerLocation = this.setMarkerLocation;
  81 + this.map.savePolygonLocation = this.savePolygonLocation;
81 this.pageLink = { 82 this.pageLink = {
82 page: 0, 83 page: 0,
83 pageSize: this.settings.mapPageSize, 84 pageSize: this.settings.mapPageSize,
@@ -239,6 +240,56 @@ export class MapWidgetController implements MapWidgetInterface { @@ -239,6 +240,56 @@ export class MapWidgetController implements MapWidgetInterface {
239 } 240 }
240 } 241 }
241 242
  243 + savePolygonLocation = (e: FormattedData, coordinates?: Array<any>) => {
  244 + const attributeService = this.ctx.$injector.get(AttributeService);
  245 +
  246 + const entityId: EntityId = {
  247 + entityType: e.$datasource.entityType,
  248 + id: e.$datasource.entityId
  249 + };
  250 + const attributes = [];
  251 + const timeseries = [];
  252 +
  253 + const coordinatesProperties = this.settings.polygonKeyName;
  254 + e.$datasource.dataKeys.forEach(key => {
  255 + let value;
  256 + if (coordinatesProperties == key.name) {
  257 + value = {
  258 + key: key.name,
  259 + value: isDefined(coordinates) ? coordinates : e[key.name]
  260 + };
  261 + }
  262 + if (value) {
  263 + if (key.type === DataKeyType.attribute) {
  264 + attributes.push(value)
  265 + }
  266 + if (key.type === DataKeyType.timeseries) {
  267 + timeseries.push(value)
  268 + }
  269 + }
  270 + });
  271 + const observables: Observable<any>[] = [];
  272 + if (timeseries.length) {
  273 + observables.push(attributeService.saveEntityTimeseries(
  274 + entityId,
  275 + LatestTelemetry.LATEST_TELEMETRY,
  276 + timeseries
  277 + ));
  278 + }
  279 + if (attributes.length) {
  280 + observables.push(attributeService.saveEntityAttributes(
  281 + entityId,
  282 + AttributeScope.SERVER_SCOPE,
  283 + attributes
  284 + ));
  285 + }
  286 + if (observables.length) {
  287 + return forkJoin(observables);
  288 + } else {
  289 + return of(null);
  290 + }
  291 + }
  292 +
242 initSettings(settings: UnitedMapSettings, isEditMap?: boolean): UnitedMapSettings { 293 initSettings(settings: UnitedMapSettings, isEditMap?: boolean): UnitedMapSettings {
243 const functionParams = ['data', 'dsData', 'dsIndex']; 294 const functionParams = ['data', 'dsData', 'dsIndex'];
244 this.provider = settings.provider || this.mapProvider; 295 this.provider = settings.provider || this.mapProvider;
@@ -270,6 +321,9 @@ export class MapWidgetController implements MapWidgetInterface { @@ -270,6 +321,9 @@ export class MapWidgetController implements MapWidgetInterface {
270 if (isEditMap && !settings.hasOwnProperty('draggableMarker')) { 321 if (isEditMap && !settings.hasOwnProperty('draggableMarker')) {
271 settings.draggableMarker = true; 322 settings.draggableMarker = true;
272 } 323 }
  324 + if (isEditMap && !settings.hasOwnProperty('editablePolygon')) {
  325 + settings.editablePolygon = true;
  326 + }
273 return { ...defaultSettings, ...settings, ...customOptions, } 327 return { ...defaultSettings, ...settings, ...customOptions, }
274 } 328 }
275 329
@@ -22,13 +22,13 @@ @@ -22,13 +22,13 @@
22 background-repeat: no-repeat; 22 background-repeat: no-repeat;
23 } 23 }
24 24
25 -.leaflet-div-icon,  
26 -.tb-marker-label,  
27 -.tb-marker-label:before {  
28 - border: none;  
29 - background: none;  
30 - box-shadow: none;  
31 -} 25 +//.leaflet-div-icon,
  26 +//.tb-marker-label,
  27 +//.tb-marker-label:before {
  28 +// border: none;
  29 +// background: none;
  30 +// box-shadow: none;
  31 +//}
32 32
33 .leaflet-container{ 33 .leaflet-container{
34 background-color: white; 34 background-color: white;
@@ -16,6 +16,7 @@ @@ -16,6 +16,7 @@
16 16
17 import L, { LatLngExpression, LeafletMouseEvent } from 'leaflet'; 17 import L, { LatLngExpression, LeafletMouseEvent } from 'leaflet';
18 import { createTooltip, functionValueCalculator, parseWithTranslation, safeExecute } from './maps-utils'; 18 import { createTooltip, functionValueCalculator, parseWithTranslation, safeExecute } from './maps-utils';
  19 +import 'leaflet-editable/src/Leaflet.Editable';
19 import { FormattedData, PolygonSettings } from './map-models'; 20 import { FormattedData, PolygonSettings } from './map-models';
20 21
21 export class Polygon { 22 export class Polygon {
@@ -25,11 +26,10 @@ export class Polygon { @@ -25,11 +26,10 @@ export class Polygon {
25 data: FormattedData; 26 data: FormattedData;
26 dataSources: FormattedData[]; 27 dataSources: FormattedData[];
27 28
28 - constructor(public map, polyData: FormattedData, dataSources: FormattedData[], private settings: PolygonSettings) { 29 + constructor(public map, polyData: FormattedData, dataSources: FormattedData[], private settings: PolygonSettings, onDragendListener?) {
29 this.dataSources = dataSources; 30 this.dataSources = dataSources;
30 this.data = polyData; 31 this.data = polyData;
31 const polygonColor = this.getPolygonColor(settings); 32 const polygonColor = this.getPolygonColor(settings);
32 -  
33 this.leafletPoly = L.polygon(polyData[this.settings.polygonKeyName], { 33 this.leafletPoly = L.polygon(polyData[this.settings.polygonKeyName], {
34 fill: true, 34 fill: true,
35 fillColor: polygonColor, 35 fillColor: polygonColor,
@@ -38,6 +38,14 @@ export class Polygon { @@ -38,6 +38,14 @@ export class Polygon {
38 fillOpacity: settings.polygonOpacity, 38 fillOpacity: settings.polygonOpacity,
39 opacity: settings.polygonStrokeOpacity 39 opacity: settings.polygonStrokeOpacity
40 }).addTo(this.map); 40 }).addTo(this.map);
  41 + if (settings.editablePolygon) {
  42 + this.leafletPoly.enableEdit(this.map);
  43 + if (onDragendListener) {
  44 + this.leafletPoly.on('editable:vertex:dragend', e => onDragendListener(e, this.data));
  45 + this.leafletPoly.on('editable:vertex:deleted', e => onDragendListener(e, this.data));
  46 + }
  47 + }
  48 +
41 49
42 if (settings.showPolygonTooltip) { 50 if (settings.showPolygonTooltip) {
43 this.tooltip = createTooltip(this.leafletPoly, settings, polyData.$datasource); 51 this.tooltip = createTooltip(this.leafletPoly, settings, polyData.$datasource);
@@ -64,7 +72,13 @@ export class Polygon { @@ -64,7 +72,13 @@ export class Polygon {
64 updatePolygon(data: FormattedData, dataSources: FormattedData[], settings: PolygonSettings) { 72 updatePolygon(data: FormattedData, dataSources: FormattedData[], settings: PolygonSettings) {
65 this.data = data; 73 this.data = data;
66 this.dataSources = dataSources; 74 this.dataSources = dataSources;
  75 + if (settings.editablePolygon) {
  76 + this.leafletPoly.disableEdit();
  77 + }
67 this.leafletPoly.setLatLngs(data[this.settings.polygonKeyName]); 78 this.leafletPoly.setLatLngs(data[this.settings.polygonKeyName]);
  79 + if (settings.editablePolygon) {
  80 + this.leafletPoly.enableEdit(this.map);
  81 + }
68 if (settings.showPolygonTooltip) 82 if (settings.showPolygonTooltip)
69 this.updateTooltip(this.data); 83 this.updateTooltip(this.data);
70 this.updatePolygonColor(settings); 84 this.updatePolygonColor(settings);
@@ -20,7 +20,6 @@ import LeafletMap from '../leaflet-map'; @@ -20,7 +20,6 @@ import LeafletMap from '../leaflet-map';
20 import { UnitedMapSettings } from '../map-models'; 20 import { UnitedMapSettings } from '../map-models';
21 import 'leaflet.gridlayer.googlemutant'; 21 import 'leaflet.gridlayer.googlemutant';
22 import { ResourcesService } from '@core/services/resources.service'; 22 import { ResourcesService } from '@core/services/resources.service';
23 -import { Injector } from '@angular/core';  
24 import { WidgetContext } from '@home/models/widget-component.models'; 23 import { WidgetContext } from '@home/models/widget-component.models';
25 24
26 const gmGlobals: GmGlobal = {}; 25 const gmGlobals: GmGlobal = {};
@@ -37,7 +36,10 @@ export class GoogleMap extends LeafletMap { @@ -37,7 +36,10 @@ export class GoogleMap extends LeafletMap {
37 this.resource = ctx.$injector.get(ResourcesService); 36 this.resource = ctx.$injector.get(ResourcesService);
38 super.initSettings(options); 37 super.initSettings(options);
39 this.loadGoogle(() => { 38 this.loadGoogle(() => {
40 - const map = L.map($container, {attributionControl: false}).setView(options?.defaultCenterPosition, options?.defaultZoomLevel); 39 + const map = L.map($container, {
  40 + attributionControl: false,
  41 + editable: !!options.editablePolygon
  42 + }).setView(options?.defaultCenterPosition, options?.defaultZoomLevel);
41 (L.gridLayer as any).googleMutant({ 43 (L.gridLayer as any).googleMutant({
42 type: options?.gmDefaultMapType || 'roadmap' 44 type: options?.gmDefaultMapType || 'roadmap'
43 }).addTo(map); 45 }).addTo(map);
@@ -22,7 +22,9 @@ import { WidgetContext } from '@home/models/widget-component.models'; @@ -22,7 +22,9 @@ import { WidgetContext } from '@home/models/widget-component.models';
22 export class HEREMap extends LeafletMap { 22 export class HEREMap extends LeafletMap {
23 constructor(ctx: WidgetContext, $container, options: UnitedMapSettings) { 23 constructor(ctx: WidgetContext, $container, options: UnitedMapSettings) {
24 super(ctx, $container, options); 24 super(ctx, $container, options);
25 - const map = L.map($container).setView(options?.defaultCenterPosition, options?.defaultZoomLevel); 25 + const map = L.map($container, {
  26 + editable: !!options.editablePolygon
  27 + }).setView(options?.defaultCenterPosition, options?.defaultZoomLevel);
26 const tileLayer = (L.tileLayer as any).provider(options.mapProviderHere || 'HERE.normalDay', options.credentials); 28 const tileLayer = (L.tileLayer as any).provider(options.mapProviderHere || 'HERE.normalDay', options.credentials);
27 tileLayer.addTo(map); 29 tileLayer.addTo(map);
28 super.initSettings(options); 30 super.initSettings(options);
@@ -196,14 +196,15 @@ export class ImageMap extends LeafletMap { @@ -196,14 +196,15 @@ export class ImageMap extends LeafletMap {
196 initMap(updateImage?: boolean) { 196 initMap(updateImage?: boolean) {
197 if (!this.map && this.aspect > 0) { 197 if (!this.map && this.aspect > 0) {
198 const center = this.pointToLatLng(this.width / 2, this.height / 2); 198 const center = this.pointToLatLng(this.width / 2, this.height / 2);
199 - this.map = L.map(this.$container, { 199 + this.map = L.map(this.$container, {
200 minZoom: 1, 200 minZoom: 1,
201 maxZoom, 201 maxZoom,
202 scrollWheelZoom: !this.options.disableScrollZooming, 202 scrollWheelZoom: !this.options.disableScrollZooming,
203 center, 203 center,
204 zoom: 1, 204 zoom: 1,
205 crs: L.CRS.Simple, 205 crs: L.CRS.Simple,
206 - attributionControl: false 206 + attributionControl: false,
  207 + editable: !!this.options.editablePolygon
207 }); 208 });
208 this.updateBounds(updateImage); 209 this.updateBounds(updateImage);
209 } 210 }
@@ -221,14 +222,17 @@ export class ImageMap extends LeafletMap { @@ -221,14 +222,17 @@ export class ImageMap extends LeafletMap {
221 expression.y * this.height); 222 expression.y * this.height);
222 } 223 }
223 224
224 - convertPositionPolygon(expression: Array<[number, number]>): L.LatLngExpression[] {  
225 - return expression.map((el) => {  
226 - if (el.length === 2 && !el.some(isNaN)) { 225 + convertPositionPolygon(expression: (LatLngTuple | LatLngTuple[] | LatLngTuple[][])[]){
  226 + return (expression).map((el) => {
  227 + if (!Array.isArray(el[0]) && !Array.isArray(el[1]) && el.length === 2) {
227 return this.pointToLatLng( 228 return this.pointToLatLng(
228 el[0] * this.width, 229 el[0] * this.width,
229 el[1] * this.height) 230 el[1] * this.height)
  231 + } else if (Array.isArray(el) && el.length) {
  232 + return this.convertPositionPolygon(el as LatLngTuple[] | LatLngTuple[][]);
  233 + } else {
  234 + return null;
230 } 235 }
231 - return null;  
232 }).filter(el => !!el) 236 }).filter(el => !!el)
233 } 237 }
234 238
@@ -247,4 +251,25 @@ export class ImageMap extends LeafletMap { @@ -247,4 +251,25 @@ export class ImageMap extends LeafletMap {
247 [this.options.yPosKeyName]: calculateNewPointCoordinate(point.y, this.height) 251 [this.options.yPosKeyName]: calculateNewPointCoordinate(point.y, this.height)
248 } 252 }
249 } 253 }
  254 +
  255 + convertToPolygonFormat(points: Array<any>): Array<any> {
  256 + if (points.length) {
  257 + return points.map(point=> {
  258 + if (point.length) {
  259 + return this.convertToPolygonFormat(point);
  260 + } else {
  261 + const pos = this.latLngToPoint(point);
  262 + return [calculateNewPointCoordinate(pos.x, this.width), calculateNewPointCoordinate(pos.y, this.height)];
  263 + }
  264 + })
  265 + } else {
  266 + return []
  267 + }
  268 + }
  269 +
  270 + convertPolygonToCustomFormat(expression: any[][]): object {
  271 + return {
  272 + [this.options.polygonKeyName] : this.convertToPolygonFormat(expression)
  273 + }
  274 + }
250 } 275 }
@@ -22,7 +22,9 @@ import { WidgetContext } from '@home/models/widget-component.models'; @@ -22,7 +22,9 @@ import { WidgetContext } from '@home/models/widget-component.models';
22 export class OpenStreetMap extends LeafletMap { 22 export class OpenStreetMap extends LeafletMap {
23 constructor(ctx: WidgetContext, $container, options: UnitedMapSettings) { 23 constructor(ctx: WidgetContext, $container, options: UnitedMapSettings) {
24 super(ctx, $container, options); 24 super(ctx, $container, options);
25 - const map = L.map($container).setView(options?.defaultCenterPosition, options?.defaultZoomLevel); 25 + const map = L.map($container, {
  26 + editable: !!options.editablePolygon
  27 + }).setView(options?.defaultCenterPosition, options?.defaultZoomLevel);
26 let tileLayer; 28 let tileLayer;
27 if (options.useCustomProvider) 29 if (options.useCustomProvider)
28 tileLayer = L.tileLayer(options.customProviderTileUrl); 30 tileLayer = L.tileLayer(options.customProviderTileUrl);
@@ -24,7 +24,9 @@ export class TencentMap extends LeafletMap { @@ -24,7 +24,9 @@ export class TencentMap extends LeafletMap {
24 constructor(ctx: WidgetContext, $container, options: UnitedMapSettings) { 24 constructor(ctx: WidgetContext, $container, options: UnitedMapSettings) {
25 super(ctx, $container, options); 25 super(ctx, $container, options);
26 const txUrl = 'http://rt{s}.map.gtimg.com/realtimerender?z={z}&x={x}&y={y}&type=vector&style=0'; 26 const txUrl = 'http://rt{s}.map.gtimg.com/realtimerender?z={z}&x={x}&y={y}&type=vector&style=0';
27 - const map = L.map($container).setView(options?.defaultCenterPosition, options?.defaultZoomLevel); 27 + const map = L.map($container, {
  28 + editable: !!options.editablePolygon
  29 + }).setView(options?.defaultCenterPosition, options?.defaultZoomLevel);
28 const txLayer = L.tileLayer(txUrl, { 30 const txLayer = L.tileLayer(txUrl, {
29 subdomains: '0123', 31 subdomains: '0123',
30 tms: true, 32 tms: true,
@@ -524,6 +524,11 @@ export const mapPolygonSchema = @@ -524,6 +524,11 @@ export const mapPolygonSchema =
524 type: 'string', 524 type: 'string',
525 default: 'coordinates' 525 default: 'coordinates'
526 }, 526 },
  527 + editablePolygon: {
  528 + title: 'Enable polygon edit',
  529 + type: 'boolean',
  530 + default: false
  531 + },
527 polygonColor: { 532 polygonColor: {
528 title: 'Polygon color', 533 title: 'Polygon color',
529 type: 'string' 534 type: 'string'
@@ -581,6 +586,7 @@ export const mapPolygonSchema = @@ -581,6 +586,7 @@ export const mapPolygonSchema =
581 form: [ 586 form: [
582 'showPolygon', 587 'showPolygon',
583 'polygonKeyName', 588 'polygonKeyName',
  589 + 'editablePolygon',
584 { 590 {
585 key: 'polygonColor', 591 key: 'polygonColor',
586 type: 'color' 592 type: 'color'
  1 +<svg xmlns="http://www.w3.org/2000/svg" width="48" height="48" viewBox="0 0 48 48">
  2 + <polygon points="2,46 46,46 24,2" style="fill:transparent;stroke:black;stroke-width:2"/>
  3 +</svg>
@@ -20,9 +20,12 @@ declare module 'leaflet' { @@ -20,9 +20,12 @@ declare module 'leaflet' {
20 20
21 namespace Control { 21 namespace Control {
22 class AddMarker extends L.Control { } 22 class AddMarker extends L.Control { }
  23 + class AddPolygon extends L.Control { }
23 } 24 }
24 25
25 namespace control { 26 namespace control {
26 function addMarker(options): Control.AddMarker; 27 function addMarker(options): Control.AddMarker;
  28 + function addPolygon(options): Control.AddPolygon;
27 } 29 }
28 -}  
  30 +
  31 +}
  1 +///
  2 +/// Copyright © 2016-2020 The Thingsboard Authors
  3 +///
  4 +/// Licensed under the Apache License, Version 2.0 (the "License");
  5 +/// you may not use this file except in compliance with the License.
  6 +/// You may obtain a copy of the License at
  7 +///
  8 +/// http://www.apache.org/licenses/LICENSE-2.0
  9 +///
  10 +/// Unless required by applicable law or agreed to in writing, software
  11 +/// distributed under the License is distributed on an "AS IS" BASIS,
  12 +/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13 +/// See the License for the specific language governing permissions and
  14 +/// limitations under the License.
  15 +///
  16 +
  17 +import * as leaflet from 'leaflet';
  18 +
  19 +declare module 'leaflet' {
  20 +
  21 + /**
  22 + * Make geometries editable in Leaflet.
  23 + *
  24 + * This is not a plug and play UI, and will not. This is a minimal, lightweight, and fully extendable API to
  25 + * control editing of geometries. So you can easily build your own UI with your own needs and choices.
  26 + */
  27 + interface EditableStatic {
  28 + new (map: Map, options: EditOptions): Editable;
  29 + }
  30 +
  31 + /**
  32 + * Options to pass to L.Editable when instanciating.
  33 + */
  34 + interface EditOptions extends leaflet.MapOptions{
  35 + /**
  36 + * Whether to create a L.Editable instance at map init or not.
  37 + */
  38 + editable: boolean;
  39 + /**
  40 + * Class to be used when creating a new Polyline.
  41 + */
  42 + polylineClass?: object;
  43 +
  44 + /**
  45 + * Class to be used when creating a new Polygon.
  46 + */
  47 + polygonClass?: object;
  48 +
  49 + /**
  50 + * Class to be used when creating a new Marker.
  51 + */
  52 + markerClass?: object;
  53 +
  54 + /**
  55 + * CSS class to be added to the map container while drawing.
  56 + */
  57 + drawingCSSClass?: string;
  58 +
  59 + /**
  60 + * Layer used to store edit tools (vertex, line guide…).
  61 + */
  62 + editLayer?: LayerGroup<leaflet.Layer>;
  63 +
  64 + /**
  65 + * Default layer used to store drawn features (marker, polyline…).
  66 + */
  67 + featuresLayer?: LayerGroup<Polyline|Polygon|Marker>;
  68 +
  69 + /**
  70 + * Class to be used as vertex, for path editing.
  71 + */
  72 + vertexMarkerClass?: object;
  73 +
  74 + /**
  75 + * Class to be used as middle vertex, pulled by the user to create a new point in the middle of a path.
  76 + */
  77 + middleMarkerClass?: object;
  78 +
  79 + /**
  80 + * Class to be used as Polyline editor.
  81 + */
  82 + polylineEditorClass?: object;
  83 +
  84 + /**
  85 + * Class to be used as Polygon editor.
  86 + */
  87 + polygonEditorClass?: object;
  88 +
  89 + /**
  90 + * Class to be used as Marker editor.
  91 + */
  92 + markerEditorClass?: object;
  93 +
  94 + /**
  95 + * Options to be passed to the line guides.
  96 + */
  97 + lineGuideOptions?: object;
  98 +
  99 + /**
  100 + * Set this to true if you don't want middle markers.
  101 + */
  102 + skipMiddleMarkers?: boolean;
  103 + }
  104 +
  105 + /**
  106 + * Make geometries editable in Leaflet.
  107 + *
  108 + * This is not a plug and play UI, and will not. This is a minimal, lightweight, and fully extendable API to
  109 + * control editing of geometries. So you can easily build your own UI with your own needs and choices.
  110 + */
  111 + interface Editable extends leaflet.Evented {
  112 + /**
  113 + * Options to pass to L.Editable when instanciating.
  114 + */
  115 + options: EditOptions;
  116 +
  117 + currentPolygon: Polyline|Polygon|Marker;
  118 +
  119 + /**
  120 + * Start drawing a polyline. If latlng is given, a first point will be added. In any case, continuing on user
  121 + * click. If options is given, it will be passed to the polyline class constructor.
  122 + */
  123 + startPolyline(latLng?: LatLng, options?: PolylineOptions): Polyline;
  124 +
  125 + /**
  126 + * Start drawing a polygon. If latlng is given, a first point will be added. In any case, continuing on user
  127 + * click. If options is given, it will be passed to the polygon class constructor.
  128 + */
  129 + startPolygon(latLng?: LatLng, options?: PolylineOptions): Polygon;
  130 +
  131 + /**
  132 + * Start adding a marker. If latlng is given, the marker will be shown first at this point. In any case, it
  133 + * will follow the user mouse, and will have a final latlng on next click (or touch). If options is given,
  134 + * it will be passed to the marker class constructor.
  135 + */
  136 + startMarker(latLng?: LatLng, options?: MarkerOptions): Marker;
  137 +
  138 + /**
  139 + * When you need to stop any ongoing drawing, without needing to know which editor is active.
  140 + */
  141 + stopDrawing(): void;
  142 +
  143 + /**
  144 + * When you need to commit any ongoing drawing, without needing to know which editor is active.
  145 + */
  146 + commitDrawing(): void;
  147 + }
  148 +
  149 + let Editable: EditableStatic;
  150 +
  151 + /**
  152 + * EditableMixin is included to L.Polyline, L.Polygon and L.Marker. It adds the following methods to them.
  153 + *
  154 + * When editing is enabled, the editor is accessible on the instance with the editor property.
  155 + */
  156 + interface EditableMixin {
  157 + /**
  158 + * Enable editing, by creating an editor if not existing, and then calling enable on it.
  159 + */
  160 + enableEdit(map: L.Map): any;
  161 +
  162 + /**
  163 + * Disable editing, also remove the editor property reference.
  164 + */
  165 + disableEdit(): void;
  166 +
  167 + /**
  168 + * Enable or disable editing, according to current status.
  169 + */
  170 + toggleEdit(): void;
  171 +
  172 + /**
  173 + * Return true if current instance has an editor attached, and this editor is enabled.
  174 + */
  175 + editEnabled(): boolean;
  176 + }
  177 +
  178 + interface Map {
  179 +
  180 +
  181 + /**
  182 + * Options to pass to L.Editable when instanciating.
  183 + */
  184 + editOptions: MapOptions;
  185 +
  186 + /**
  187 + * L.Editable plugin instance.
  188 + */
  189 + editTools: Editable;
  190 + }
  191 +
  192 + // tslint:disable-next-line:no-empty-interface
  193 + interface Polyline extends EditableMixin {}
  194 +
  195 + namespace Map {
  196 + interface MapOptions {
  197 + /**
  198 + * Whether to create a L.Editable instance at map init or not.
  199 + */
  200 + editable?: boolean;
  201 +
  202 + /**
  203 + * Options to pass to L.Editable when instanciating.
  204 + */
  205 + editOptions?: MapOptions;
  206 + }
  207 + }
  208 +
  209 + /**
  210 + * When editing a feature (marker, polyline…), an editor is attached to it. This editor basically knows
  211 + * how to handle the edition.
  212 + */
  213 + interface BaseEditor {
  214 + /**
  215 + * Set up the drawing tools for the feature to be editable.
  216 + */
  217 + enable(): MarkerEditor|PolylineEditor|PolygonEditor;
  218 +
  219 + /**
  220 + * Remove editing tools.
  221 + */
  222 + disable(): MarkerEditor|PolylineEditor|PolygonEditor;
  223 + }
  224 +
  225 + /**
  226 + * Inherit from L.Editable.BaseEditor.
  227 + * Inherited by L.Editable.PolylineEditor and L.Editable.PolygonEditor.
  228 + */
  229 + interface PathEditor extends BaseEditor {
  230 + /**
  231 + * Rebuild edit elements (vertex, middlemarker, etc.).
  232 + */
  233 + reset(): void;
  234 + }
  235 +
  236 + /**
  237 + * Inherit from L.Editable.PathEditor.
  238 + */
  239 + interface PolylineEditor extends PathEditor {
  240 + /**
  241 + * Set up drawing tools to continue the line forward.
  242 + */
  243 + continueForward(): void;
  244 +
  245 + /**
  246 + * Set up drawing tools to continue the line backward.
  247 + */
  248 + continueBackward(): void;
  249 + }
  250 +
  251 + /**
  252 + * Inherit from L.Editable.PathEditor.
  253 + */
  254 + interface PolygonEditor extends PathEditor {
  255 + /**
  256 + * Set up drawing tools for creating a new hole on the polygon. If the latlng param is given, a first
  257 + * point is created.
  258 + */
  259 + newHole(latlng: LatLng): void;
  260 + }
  261 +
  262 + /**
  263 + * Inherit from L.Editable.BaseEditor.
  264 + */
  265 + // tslint:disable-next-line:no-empty-interface
  266 + interface MarkerEditor extends BaseEditor {}
  267 +
  268 + interface Marker extends EditableMixin, MarkerEditor {}
  269 +
  270 + interface Polyline extends EditableMixin, PolylineEditor {}
  271 +
  272 + interface Polygon extends EditableMixin, PolygonEditor {}
  273 +
  274 + function map(element: string | HTMLElement, options?: EditOptions): Map;
  275 +}
@@ -20,7 +20,8 @@ @@ -20,7 +20,8 @@
20 "src/typings/jquery.flot.typings.d.ts", 20 "src/typings/jquery.flot.typings.d.ts",
21 "src/typings/jquery.jstree.typings.d.ts", 21 "src/typings/jquery.jstree.typings.d.ts",
22 "src/typings/split.js.typings.d.ts", 22 "src/typings/split.js.typings.d.ts",
23 - "src/typings/add-marker.d.ts" 23 + "src/typings/add-marker.d.ts",
  24 + "src/typings/leaflet-editable.d.ts"
24 ], 25 ],
25 "paths": { 26 "paths": {
26 "@app/*": ["src/app/*"], 27 "@app/*": ["src/app/*"],