In addition to using const and let, this also upgrades our linter config and removes lint (mostly whitespace).
291 lines
6.4 KiB
JavaScript
291 lines
6.4 KiB
JavaScript
import Map from '../src/ol/Map.js';
|
|
import Observable from '../src/ol/Observable.js';
|
|
import Overlay from '../src/ol/Overlay.js';
|
|
import {getArea, getLength} from '../src/ol/sphere.js';
|
|
import View from '../src/ol/View.js';
|
|
import LineString from '../src/ol/geom/LineString.js';
|
|
import Polygon from '../src/ol/geom/Polygon.js';
|
|
import Draw from '../src/ol/interaction/Draw.js';
|
|
import TileLayer from '../src/ol/layer/Tile.js';
|
|
import VectorLayer from '../src/ol/layer/Vector.js';
|
|
import OSM from '../src/ol/source/OSM.js';
|
|
import VectorSource from '../src/ol/source/Vector.js';
|
|
import CircleStyle from '../src/ol/style/Circle.js';
|
|
import Fill from '../src/ol/style/Fill.js';
|
|
import Stroke from '../src/ol/style/Stroke.js';
|
|
import Style from '../src/ol/style/Style.js';
|
|
|
|
|
|
const raster = new TileLayer({
|
|
source: new OSM()
|
|
});
|
|
|
|
const source = new VectorSource();
|
|
|
|
const vector = new VectorLayer({
|
|
source: source,
|
|
style: new Style({
|
|
fill: new Fill({
|
|
color: 'rgba(255, 255, 255, 0.2)'
|
|
}),
|
|
stroke: new Stroke({
|
|
color: '#ffcc33',
|
|
width: 2
|
|
}),
|
|
image: new CircleStyle({
|
|
radius: 7,
|
|
fill: new Fill({
|
|
color: '#ffcc33'
|
|
})
|
|
})
|
|
})
|
|
});
|
|
|
|
|
|
/**
|
|
* Currently drawn feature.
|
|
* @type {ol.Feature}
|
|
*/
|
|
let sketch;
|
|
|
|
|
|
/**
|
|
* The help tooltip element.
|
|
* @type {Element}
|
|
*/
|
|
let helpTooltipElement;
|
|
|
|
|
|
/**
|
|
* Overlay to show the help messages.
|
|
* @type {ol.Overlay}
|
|
*/
|
|
let helpTooltip;
|
|
|
|
|
|
/**
|
|
* The measure tooltip element.
|
|
* @type {Element}
|
|
*/
|
|
let measureTooltipElement;
|
|
|
|
|
|
/**
|
|
* Overlay to show the measurement.
|
|
* @type {ol.Overlay}
|
|
*/
|
|
let measureTooltip;
|
|
|
|
|
|
/**
|
|
* Message to show when the user is drawing a polygon.
|
|
* @type {string}
|
|
*/
|
|
const continuePolygonMsg = 'Click to continue drawing the polygon';
|
|
|
|
|
|
/**
|
|
* Message to show when the user is drawing a line.
|
|
* @type {string}
|
|
*/
|
|
const continueLineMsg = 'Click to continue drawing the line';
|
|
|
|
|
|
/**
|
|
* Handle pointer move.
|
|
* @param {ol.MapBrowserEvent} evt The event.
|
|
*/
|
|
const pointerMoveHandler = function(evt) {
|
|
if (evt.dragging) {
|
|
return;
|
|
}
|
|
/** @type {string} */
|
|
let helpMsg = 'Click to start drawing';
|
|
|
|
if (sketch) {
|
|
const geom = (sketch.getGeometry());
|
|
if (geom instanceof Polygon) {
|
|
helpMsg = continuePolygonMsg;
|
|
} else if (geom instanceof LineString) {
|
|
helpMsg = continueLineMsg;
|
|
}
|
|
}
|
|
|
|
helpTooltipElement.innerHTML = helpMsg;
|
|
helpTooltip.setPosition(evt.coordinate);
|
|
|
|
helpTooltipElement.classList.remove('hidden');
|
|
};
|
|
|
|
|
|
const map = new Map({
|
|
layers: [raster, vector],
|
|
target: 'map',
|
|
view: new View({
|
|
center: [-11000000, 4600000],
|
|
zoom: 15
|
|
})
|
|
});
|
|
|
|
map.on('pointermove', pointerMoveHandler);
|
|
|
|
map.getViewport().addEventListener('mouseout', function() {
|
|
helpTooltipElement.classList.add('hidden');
|
|
});
|
|
|
|
const typeSelect = document.getElementById('type');
|
|
|
|
let draw; // global so we can remove it later
|
|
|
|
|
|
/**
|
|
* Format length output.
|
|
* @param {ol.geom.LineString} line The line.
|
|
* @return {string} The formatted length.
|
|
*/
|
|
const formatLength = function(line) {
|
|
const length = getLength(line);
|
|
let output;
|
|
if (length > 100) {
|
|
output = (Math.round(length / 1000 * 100) / 100) +
|
|
' ' + 'km';
|
|
} else {
|
|
output = (Math.round(length * 100) / 100) +
|
|
' ' + 'm';
|
|
}
|
|
return output;
|
|
};
|
|
|
|
|
|
/**
|
|
* Format area output.
|
|
* @param {ol.geom.Polygon} polygon The polygon.
|
|
* @return {string} Formatted area.
|
|
*/
|
|
const formatArea = function(polygon) {
|
|
const area = getArea(polygon);
|
|
let output;
|
|
if (area > 10000) {
|
|
output = (Math.round(area / 1000000 * 100) / 100) +
|
|
' ' + 'km<sup>2</sup>';
|
|
} else {
|
|
output = (Math.round(area * 100) / 100) +
|
|
' ' + 'm<sup>2</sup>';
|
|
}
|
|
return output;
|
|
};
|
|
|
|
function addInteraction() {
|
|
const type = (typeSelect.value == 'area' ? 'Polygon' : 'LineString');
|
|
draw = new Draw({
|
|
source: source,
|
|
type: type,
|
|
style: new Style({
|
|
fill: new Fill({
|
|
color: 'rgba(255, 255, 255, 0.2)'
|
|
}),
|
|
stroke: new Stroke({
|
|
color: 'rgba(0, 0, 0, 0.5)',
|
|
lineDash: [10, 10],
|
|
width: 2
|
|
}),
|
|
image: new CircleStyle({
|
|
radius: 5,
|
|
stroke: new Stroke({
|
|
color: 'rgba(0, 0, 0, 0.7)'
|
|
}),
|
|
fill: new Fill({
|
|
color: 'rgba(255, 255, 255, 0.2)'
|
|
})
|
|
})
|
|
})
|
|
});
|
|
map.addInteraction(draw);
|
|
|
|
createMeasureTooltip();
|
|
createHelpTooltip();
|
|
|
|
let listener;
|
|
draw.on('drawstart',
|
|
function(evt) {
|
|
// set sketch
|
|
sketch = evt.feature;
|
|
|
|
/** @type {ol.Coordinate|undefined} */
|
|
let tooltipCoord = evt.coordinate;
|
|
|
|
listener = sketch.getGeometry().on('change', function(evt) {
|
|
const geom = evt.target;
|
|
let output;
|
|
if (geom instanceof Polygon) {
|
|
output = formatArea(geom);
|
|
tooltipCoord = geom.getInteriorPoint().getCoordinates();
|
|
} else if (geom instanceof LineString) {
|
|
output = formatLength(geom);
|
|
tooltipCoord = geom.getLastCoordinate();
|
|
}
|
|
measureTooltipElement.innerHTML = output;
|
|
measureTooltip.setPosition(tooltipCoord);
|
|
});
|
|
}, this);
|
|
|
|
draw.on('drawend',
|
|
function() {
|
|
measureTooltipElement.className = 'tooltip tooltip-static';
|
|
measureTooltip.setOffset([0, -7]);
|
|
// unset sketch
|
|
sketch = null;
|
|
// unset tooltip so that a new one can be created
|
|
measureTooltipElement = null;
|
|
createMeasureTooltip();
|
|
Observable.unByKey(listener);
|
|
}, this);
|
|
}
|
|
|
|
|
|
/**
|
|
* Creates a new help tooltip
|
|
*/
|
|
function createHelpTooltip() {
|
|
if (helpTooltipElement) {
|
|
helpTooltipElement.parentNode.removeChild(helpTooltipElement);
|
|
}
|
|
helpTooltipElement = document.createElement('div');
|
|
helpTooltipElement.className = 'tooltip hidden';
|
|
helpTooltip = new Overlay({
|
|
element: helpTooltipElement,
|
|
offset: [15, 0],
|
|
positioning: 'center-left'
|
|
});
|
|
map.addOverlay(helpTooltip);
|
|
}
|
|
|
|
|
|
/**
|
|
* Creates a new measure tooltip
|
|
*/
|
|
function createMeasureTooltip() {
|
|
if (measureTooltipElement) {
|
|
measureTooltipElement.parentNode.removeChild(measureTooltipElement);
|
|
}
|
|
measureTooltipElement = document.createElement('div');
|
|
measureTooltipElement.className = 'tooltip tooltip-measure';
|
|
measureTooltip = new Overlay({
|
|
element: measureTooltipElement,
|
|
offset: [0, -15],
|
|
positioning: 'bottom-center'
|
|
});
|
|
map.addOverlay(measureTooltip);
|
|
}
|
|
|
|
|
|
/**
|
|
* Let user change the geometry type.
|
|
*/
|
|
typeSelect.onchange = function() {
|
|
map.removeInteraction(draw);
|
|
addInteraction();
|
|
};
|
|
|
|
addInteraction();
|