Skip to content
This repository has been archived by the owner on Jan 25, 2024. It is now read-only.

Commit

Permalink
Add support for last position marker (#30)
Browse files Browse the repository at this point in the history
* + timeseries support & empty point when no data

* ~ options for center pos + ant-marker type

* + tooltip from data support

* ~ build

* ~ build

* ~ build

* add last marker icon + logic

* Refactor create icon function

* Hide custom center coords when using last position

Co-authored-by: Andrey Litvinov <[email protected]>
Co-authored-by: Jack Ord <[email protected]>
  • Loading branch information
3 people authored Nov 5, 2020
1 parent c1752bc commit b5c2e64
Show file tree
Hide file tree
Showing 12 changed files with 233 additions and 48 deletions.
6 changes: 6 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -13,3 +13,9 @@ Initial Release

## v2.0.2
* Add support for tooltips on markers

## v2.0.3
* Add support for only showing last marker
* Add support for switching between primary and secondary marker icons
* Add support for changing size of the last marker
* Add support for centering map by last position
7 changes: 7 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,10 @@ Switch between views (Markers, Ant Path, Ant Path With Markers, Hexbin, Heatmap)

#### Markers
- `Size`: The size of the markers
- `Size of last marker`: The size of the last marker
- `Show only last marker`: Shows only last marker
- `Use secondary icon for last marker`: Uses secondary icon image for last marker
- `Use secondary icon for all markers`: Uses secondary icon image for all markers

![markers_options](img/markers.png)

Expand Down Expand Up @@ -70,6 +74,9 @@ Switch between views (Markers, Ant Path, Ant Path With Markers, Hexbin, Heatmap)

![heatmap_options](img/heatmap.png)

### Changing marker icons
To change the icons used for the markers, replace the `marker.png` and `marker_secondary.png` files in the `/src/img` folder.

### Updating query based on map bounds
To update the query dynamically based on the map bounds turn on `Use map bounds in query`. To use this you must manually add four variables to the dashboard (via settings in the top right corner). Add four variables of type `constant` with names `minLat`, `minLon`, `maxLat`, and `maxLon`. The values can be anything, e.g. 1, 2, 3, 4 - they will be overwritten by the plugin. Remember to save the dashboard. The variables can then be used in a query. Here is an example that limits the query to the bounds of the map:

Expand Down
7 changes: 7 additions & 0 deletions dist/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,10 @@ Switch between views (Markers, Ant Path, Ant Path With Markers, Hexbin, Heatmap)

#### Markers
- `Size`: The size of the markers
- `Size of last marker`: The size of the last marker
- `Show only last marker`: Shows only last marker
- `Use secondary icon for last marker`: Uses secondary icon image for last marker
- `Use secondary icon for all markers`: Uses secondary icon image for all markers

![markers_options](img/markers.png)

Expand Down Expand Up @@ -70,6 +74,9 @@ Switch between views (Markers, Ant Path, Ant Path With Markers, Hexbin, Heatmap)

![heatmap_options](img/heatmap.png)

### Changing marker icons
To change the icons used for the markers, replace the `marker.png` and `marker_secondary.png` files in the `/src/img` folder.

### Updating query based on map bounds
To update the query dynamically based on the map bounds turn on `Use map bounds in query`. To use this you must manually add four variables to the dashboard (via settings in the top right corner). Add four variables of type `constant` with names `minLat`, `minLon`, `maxLat`, and `maxLon`. The values can be anything, e.g. 1, 2, 3, 4 - they will be overwritten by the plugin. Remember to save the dashboard. The variables can then be used in a query. Here is an example that limits the query to the bounds of the map:

Expand Down
Binary file added dist/img/marker_secondary.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
130 changes: 108 additions & 22 deletions dist/module.js
Original file line number Diff line number Diff line change
Expand Up @@ -27929,10 +27929,11 @@ var TrackMapPanel = function TrackMapPanel(_a) {
var styles = getStyles();
var mapRef = Object(react__WEBPACK_IMPORTED_MODULE_1__["useRef"])(null);
var WrappedHexbinLayer = Object(react_leaflet__WEBPACK_IMPORTED_MODULE_5__["withLeaflet"])(HexbinLayer);
var mark = new leaflet__WEBPACK_IMPORTED_MODULE_6__["Icon"]({
iconUrl: __webpack_require__(/*! img/marker.png */ "./img/marker.png"),
iconSize: [options.marker.size, options.marker.size]
});

var primaryIcon = __webpack_require__(/*! img/marker.png */ "./img/marker.png");

var secondaryIcon = __webpack_require__(/*! img/marker_secondary.png */ "./img/marker_secondary.png");

Object(react__WEBPACK_IMPORTED_MODULE_1__["useEffect"])(function () {
if (mapRef.current !== null) {
var bounds = mapRef.current.leafletElement.getBounds();
Expand Down Expand Up @@ -28020,14 +28021,41 @@ var TrackMapPanel = function TrackMapPanel(_a) {
}
});
});
var markers = [];
(_z = positions) === null || _z === void 0 ? void 0 : _z.forEach(function (p, i) {
markers.push(react__WEBPACK_IMPORTED_MODULE_1___default.a.createElement(react_leaflet__WEBPACK_IMPORTED_MODULE_5__["Marker"], {
key: i,
position: [p.latitude, p.longitude],
icon: mark
}, react__WEBPACK_IMPORTED_MODULE_1___default.a.createElement(react_leaflet__WEBPACK_IMPORTED_MODULE_5__["Popup"], null, p.tooltip)));
});

var createMarkers = function createMarkers(positions, useSecondaryIconForAllMarkers, useSecondaryIconForLastMarker, showOnlyLastMarker) {
var _a;

var markers = [];

if (((_a = positions) === null || _a === void 0 ? void 0 : _a.length) > 0) {
positions.forEach(function (p, i) {
var _a;

var isLastPosition = i + 1 === ((_a = positions) === null || _a === void 0 ? void 0 : _a.length);
var useSecondaryIcon = useSecondaryIconForAllMarkers || useSecondaryIconForLastMarker && isLastPosition;
var icon = createIcon(useSecondaryIcon ? secondaryIcon : primaryIcon, isLastPosition ? options.marker.sizeLast : options.marker.size);
markers.push(react__WEBPACK_IMPORTED_MODULE_1___default.a.createElement(react_leaflet__WEBPACK_IMPORTED_MODULE_5__["Marker"], {
key: i,
position: [p.latitude, p.longitude],
icon: icon,
title: p.tooltip
}, react__WEBPACK_IMPORTED_MODULE_1___default.a.createElement(react_leaflet__WEBPACK_IMPORTED_MODULE_5__["Popup"], null, p.tooltip)));
});
}

return showOnlyLastMarker ? [markers[markers.length - 1]] : markers;
};

var createIcon = function createIcon(url, size) {
return new leaflet__WEBPACK_IMPORTED_MODULE_6__["Icon"]({
iconUrl: url,
iconSize: [size, size],
iconAnchor: [size * 0.5, size],
popupAnchor: [0, -size]
});
};

var markers = createMarkers(positions, options.marker.useSecondaryIconForAllMarkers, options.marker.useSecondaryIconForLastMarker, options.marker.showOnlyLastMarker);
var antOptions = {
delay: options.ant.delay,
dashArray: [10, 20],
Expand Down Expand Up @@ -28086,11 +28114,23 @@ var TrackMapPanel = function TrackMapPanel(_a) {
longitude: options.map.centerLongitude
};

if (options.map.useCenterFromFirstPos && ((_0 = positions) === null || _0 === void 0 ? void 0 : _0.length) && positions[0].latitude) {
if (options.map.useCenterFromFirstPos && ((_z = positions) === null || _z === void 0 ? void 0 : _z.length) && positions[0].latitude) {
mapCenter.latitude = positions[0].latitude;
mapCenter.longitude = positions[0].longitude;
}

if ((_0 = positions) === null || _0 === void 0 ? void 0 : _0.length) {
if (options.map.useCenterFromFirstPos && positions[0].latitude) {
mapCenter.latitude = positions[0].latitude;
mapCenter.longitude = positions[0].longitude;
}

if (!options.map.useCenterFromFirstPos && options.map.useCenterFromLastPos && positions[positions.length - 1].latitude) {
mapCenter.latitude = positions[positions.length - 1].latitude;
mapCenter.longitude = positions[positions.length - 1].longitude;
}
}

return react__WEBPACK_IMPORTED_MODULE_1___default.a.createElement("div", {
className: Object(emotion__WEBPACK_IMPORTED_MODULE_3__["cx"])(styles.wrapper, Object(emotion__WEBPACK_IMPORTED_MODULE_3__["css"])(templateObject_1 || (templateObject_1 = Object(tslib__WEBPACK_IMPORTED_MODULE_0__["__makeTemplateObject"])(["\n width: ", "px;\n height: ", "px;\n "], ["\n width: ", "px;\n height: ", "px;\n "])), width, height))
}, react__WEBPACK_IMPORTED_MODULE_1___default.a.createElement(react_leaflet__WEBPACK_IMPORTED_MODULE_5__["Map"], {
Expand Down Expand Up @@ -28146,6 +28186,17 @@ module.exports = "

/***/ }),

/***/ "./img/marker_secondary.png":
/*!**********************************!*\
!*** ./img/marker_secondary.png ***!
\**********************************/
/*! no static exports found */
/***/ (function(module, exports) {

module.exports = ""

/***/ }),

/***/ "./leaflet.css":
/*!*********************!*\
!*** ./leaflet.css ***!
Expand Down Expand Up @@ -28196,19 +28247,26 @@ var plugin = new _grafana_data__WEBPACK_IMPORTED_MODULE_0__["PanelPlugin"](_Trac
path: 'map.useCenterFromFirstPos',
name: 'Map center to first position',
defaultValue: false
}).addBooleanSwitch({
path: 'map.useCenterFromLastPos',
name: 'Map center to last position',
defaultValue: false,
showIf: function showIf(config) {
return !config.map.useCenterFromFirstPos;
}
}).addNumberInput({
path: 'map.centerLatitude',
name: 'Map center latitude',
defaultValue: 56.17203,
showIf: function showIf(config) {
return !config.map.useCenterFromFirstPos;
return !config.map.useCenterFromFirstPos && !config.map.useCenterFromLastPos;
}
}).addNumberInput({
path: 'map.centerLongitude',
name: 'Map center longitude',
defaultValue: 10.1865203,
showIf: function showIf(config) {
return !config.map.useCenterFromFirstPos;
return !config.map.useCenterFromFirstPos && !config.map.useCenterFromLastPos;
}
}).addNumberInput({
path: 'map.zoom',
Expand Down Expand Up @@ -28246,42 +28304,42 @@ var plugin = new _grafana_data__WEBPACK_IMPORTED_MODULE_0__["PanelPlugin"](_Trac
name: 'Delay',
defaultValue: 400,
showIf: function showIf(config) {
return config.viewType === 'ant';
return config.viewType === 'ant' || config.viewType === 'ant-marker';
}
}).addNumberInput({
path: 'ant.weight',
name: 'Weight',
defaultValue: 5,
showIf: function showIf(config) {
return config.viewType === 'ant';
return config.viewType === 'ant' || config.viewType === 'ant-marker';
}
}).addColorPicker({
path: 'ant.color',
name: 'Color',
defaultValue: 'rgb(0, 100, 255)',
showIf: function showIf(config) {
return config.viewType === 'ant';
return config.viewType === 'ant' || config.viewType === 'ant-marker';
}
}).addColorPicker({
path: 'ant.pulseColor',
name: 'Pulse color',
defaultValue: 'rgb(0, 0, 0)',
showIf: function showIf(config) {
return config.viewType === 'ant';
return config.viewType === 'ant' || config.viewType === 'ant-marker';
}
}).addBooleanSwitch({
path: 'ant.paused',
name: 'Paused',
defaultValue: false,
showIf: function showIf(config) {
return config.viewType === 'ant';
return config.viewType === 'ant' || config.viewType === 'ant-marker';
}
}).addBooleanSwitch({
path: 'ant.reverse',
name: 'Reverse',
defaultValue: false,
showIf: function showIf(config) {
return config.viewType === 'ant';
return config.viewType === 'ant' || config.viewType === 'ant-marker';
}
}) //heat
.addBooleanSwitch({
Expand All @@ -28304,7 +28362,35 @@ var plugin = new _grafana_data__WEBPACK_IMPORTED_MODULE_0__["PanelPlugin"](_Trac
name: 'Size',
defaultValue: 25,
showIf: function showIf(config) {
return config.viewType === 'marker';
return config.viewType === 'marker' || config.viewType === 'ant-marker';
}
}).addNumberInput({
path: 'marker.sizeLast',
name: 'Size of last marker',
defaultValue: 25,
showIf: function showIf(config) {
return config.viewType === 'marker' || config.viewType === 'ant-marker';
}
}).addBooleanSwitch({
path: 'marker.showOnlyLastMarker',
name: 'Show only last marker',
defaultValue: false,
showIf: function showIf(config) {
return config.viewType === 'marker' || config.viewType === 'ant-marker';
}
}).addBooleanSwitch({
path: 'marker.useSecondaryIconForLastMarker',
name: 'Use secondary icon for last marker',
defaultValue: false,
showIf: function showIf(config) {
return config.viewType === 'marker' || config.viewType === 'ant-marker';
}
}).addBooleanSwitch({
path: 'marker.useSecondaryIconForAllMarkers',
name: 'Use secondary icon for all markers',
defaultValue: false,
showIf: function showIf(config) {
return config.viewType === 'marker' || config.viewType === 'ant-marker';
}
}) //hex
.addNumberInput({
Expand Down
2 changes: 1 addition & 1 deletion dist/module.js.map

Large diffs are not rendered by default.

4 changes: 2 additions & 2 deletions dist/plugin.json
Original file line number Diff line number Diff line change
Expand Up @@ -26,8 +26,8 @@
{"name": "MIT License", "url": "https://github.com/alexandrainst/alexandra-trackmap-panel/blob/master/LICENSE"}
],
"screenshots": [],
"version": "2.0.2",
"updated": "2020-11-04"
"version": "2.0.3",
"updated": "2020-11-05"
},
"dependencies": {
"grafanaVersion": "7.x.x",
Expand Down
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "alexandra-trackmap-panel",
"version": "2.0.2",
"version": "2.0.3",
"description": "A map plugin to visualise GPS data as markers, hexbin, ant path, or heatmap",
"scripts": {
"build": "grafana-toolkit plugin:build",
Expand Down
69 changes: 57 additions & 12 deletions src/TrackMapPanel.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -23,10 +23,8 @@ export const TrackMapPanel: React.FC<Props> = ({ options, data, width, height })

const WrappedHexbinLayer: any = withLeaflet(HexbinLayer);

const mark = new Icon({
iconUrl: require('img/marker.png'),
iconSize: [options.marker.size, options.marker.size],
});
const primaryIcon: string = require('img/marker.png');
const secondaryIcon: string = require('img/marker_secondary.png');

useEffect(() => {
if (mapRef.current !== null) {
Expand Down Expand Up @@ -106,14 +104,46 @@ export const TrackMapPanel: React.FC<Props> = ({ options, data, width, height })
} as Feature);
});

const markers: ReactElement[] = [];
positions?.forEach((p, i) => {
markers.push(
<Marker key={i} position={[p.latitude, p.longitude]} icon={mark}>
<Popup>{p.tooltip}</Popup>
</Marker>
);
});
const createMarkers = (
positions: Position[],
useSecondaryIconForAllMarkers: boolean,
useSecondaryIconForLastMarker: boolean,
showOnlyLastMarker: boolean
): ReactElement[] => {
let markers: ReactElement[] = [];
if (positions?.length > 0) {
positions.forEach((p, i) => {
const isLastPosition = i + 1 === positions?.length;
const useSecondaryIcon = useSecondaryIconForAllMarkers || (useSecondaryIconForLastMarker && isLastPosition);
const icon: Icon = createIcon(
useSecondaryIcon ? secondaryIcon : primaryIcon,
isLastPosition ? options.marker.sizeLast : options.marker.size
);
markers.push(
<Marker key={i} position={[p.latitude, p.longitude]} icon={icon} title={p.tooltip}>
<Popup>{p.tooltip}</Popup>
</Marker>
);
});
}
return showOnlyLastMarker ? [markers[markers.length - 1]] : markers;
};

const createIcon = (url: string, size: number) => {
return new Icon({
iconUrl: url,
iconSize: [size, size],
iconAnchor: [size * 0.5, size],
popupAnchor: [0, -size],
});
};

let markers: ReactElement[] = createMarkers(
positions,
options.marker.useSecondaryIconForAllMarkers,
options.marker.useSecondaryIconForLastMarker,
options.marker.showOnlyLastMarker
);

const antOptions = {
delay: options.ant.delay,
Expand Down Expand Up @@ -176,6 +206,21 @@ export const TrackMapPanel: React.FC<Props> = ({ options, data, width, height })
mapCenter.longitude = positions[0].longitude;
}

if (positions?.length) {
if (options.map.useCenterFromFirstPos && positions[0].latitude) {
mapCenter.latitude = positions[0].latitude;
mapCenter.longitude = positions[0].longitude;
}
if (
!options.map.useCenterFromFirstPos &&
options.map.useCenterFromLastPos &&
positions[positions.length - 1].latitude
) {
mapCenter.latitude = positions[positions.length - 1].latitude;
mapCenter.longitude = positions[positions.length - 1].longitude;
}
}

return (
<div
className={cx(
Expand Down
Binary file added src/img/marker_secondary.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading

0 comments on commit b5c2e64

Please sign in to comment.