import _ol_Geolocation_ from '../src/ol/Geolocation.js'; import _ol_Map_ from '../src/ol/Map.js'; import _ol_Overlay_ from '../src/ol/Overlay.js'; import _ol_View_ from '../src/ol/View.js'; import _ol_control_ from '../src/ol/control.js'; import LineString from '../src/ol/geom/LineString.js'; import _ol_layer_Tile_ from '../src/ol/layer/Tile.js'; import {fromLonLat} from '../src/ol/proj.js'; import _ol_source_OSM_ from '../src/ol/source/OSM.js'; // creating the view var view = new _ol_View_({ center: fromLonLat([5.8713, 45.6452]), zoom: 19 }); // creating the map var map = new _ol_Map_({ layers: [ new _ol_layer_Tile_({ source: new _ol_source_OSM_() }) ], target: 'map', controls: _ol_control_.defaults({ attributionOptions: { collapsible: false } }), view: view }); // Geolocation marker var markerEl = document.getElementById('geolocation_marker'); var marker = new _ol_Overlay_({ positioning: 'center-center', element: markerEl, stopEvent: false }); map.addOverlay(marker); // LineString to store the different geolocation positions. This LineString // is time aware. // The Z dimension is actually used to store the rotation (heading). var positions = new LineString([], /** @type {ol.geom.GeometryLayout} */ ('XYZM')); // Geolocation Control var geolocation = new _ol_Geolocation_({ projection: view.getProjection(), trackingOptions: { maximumAge: 10000, enableHighAccuracy: true, timeout: 600000 } }); var deltaMean = 500; // the geolocation sampling period mean in ms // Listen to position changes geolocation.on('change', function() { var position = geolocation.getPosition(); var accuracy = geolocation.getAccuracy(); var heading = geolocation.getHeading() || 0; var speed = geolocation.getSpeed() || 0; var m = Date.now(); addPosition(position, heading, m, speed); var coords = positions.getCoordinates(); var len = coords.length; if (len >= 2) { deltaMean = (coords[len - 1][3] - coords[0][3]) / (len - 1); } var html = [ 'Position: ' + position[0].toFixed(2) + ', ' + position[1].toFixed(2), 'Accuracy: ' + accuracy, 'Heading: ' + Math.round(radToDeg(heading)) + '°', 'Speed: ' + (speed * 3.6).toFixed(1) + ' km/h', 'Delta: ' + Math.round(deltaMean) + 'ms' ].join('
'); document.getElementById('info').innerHTML = html; }); geolocation.on('error', function() { alert('geolocation error'); // FIXME we should remove the coordinates in positions }); // convert radians to degrees function radToDeg(rad) { return rad * 360 / (Math.PI * 2); } // convert degrees to radians function degToRad(deg) { return deg * Math.PI * 2 / 360; } // modulo for negative values function mod(n) { return ((n % (2 * Math.PI)) + (2 * Math.PI)) % (2 * Math.PI); } function addPosition(position, heading, m, speed) { var x = position[0]; var y = position[1]; var fCoords = positions.getCoordinates(); var previous = fCoords[fCoords.length - 1]; var prevHeading = previous && previous[2]; if (prevHeading) { var headingDiff = heading - mod(prevHeading); // force the rotation change to be less than 180° if (Math.abs(headingDiff) > Math.PI) { var sign = (headingDiff >= 0) ? 1 : -1; headingDiff = -sign * (2 * Math.PI - Math.abs(headingDiff)); } heading = prevHeading + headingDiff; } positions.appendCoordinate([x, y, heading, m]); // only keep the 20 last coordinates positions.setCoordinates(positions.getCoordinates().slice(-20)); // FIXME use speed instead if (heading && speed) { markerEl.src = 'data/geolocation_marker_heading.png'; } else { markerEl.src = 'data/geolocation_marker.png'; } } // recenters the view by putting the given coordinates at 3/4 from the top or // the screen function getCenterWithHeading(position, rotation, resolution) { var size = map.getSize(); var height = size[1]; return [ position[0] - Math.sin(rotation) * height * resolution * 1 / 4, position[1] + Math.cos(rotation) * height * resolution * 1 / 4 ]; } var previousM = 0; function updateView() { // use sampling period to get a smooth transition var m = Date.now() - deltaMean * 1.5; m = Math.max(m, previousM); previousM = m; // interpolate position along positions LineString var c = positions.getCoordinateAtM(m, true); if (c) { view.setCenter(getCenterWithHeading(c, -c[2], view.getResolution())); view.setRotation(-c[2]); marker.setPosition(c); } } // geolocate device var geolocateBtn = document.getElementById('geolocate'); geolocateBtn.addEventListener('click', function() { geolocation.setTracking(true); // Start position tracking map.on('postcompose', updateView); map.render(); disableButtons(); }, false); // simulate device move var simulationData; var client = new XMLHttpRequest(); client.open('GET', 'data/geolocation-orientation.json'); /** * Handle data loading. */ client.onload = function() { simulationData = JSON.parse(client.responseText).data; }; client.send(); var simulateBtn = document.getElementById('simulate'); simulateBtn.addEventListener('click', function() { var coordinates = simulationData; var first = coordinates.shift(); simulatePositionChange(first); var prevDate = first.timestamp; function geolocate() { var position = coordinates.shift(); if (!position) { return; } var newDate = position.timestamp; simulatePositionChange(position); window.setTimeout(function() { prevDate = newDate; geolocate(); }, (newDate - prevDate) / 0.5); } geolocate(); map.on('postcompose', updateView); map.render(); disableButtons(); }, false); function simulatePositionChange(position) { var coords = position.coords; geolocation.set('accuracy', coords.accuracy); geolocation.set('heading', degToRad(coords.heading)); var projectedPosition = fromLonLat([coords.longitude, coords.latitude]); geolocation.set('position', projectedPosition); geolocation.set('speed', coords.speed); geolocation.changed(); } function disableButtons() { geolocateBtn.disabled = 'disabled'; simulateBtn.disabled = 'disabled'; }