/*
* μlogger
*
* Copyright(C) 2019 Bartek Fabiszewski (www.fabiszewski.net)
*
* This is free software; you can redistribute it and/or modify it under
* the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, see .
*/
import { lang as $, auth, config } from './initializer.js';
import GoogleMapsApi from './mapapi/api_gmaps.js';
import OpenLayersApi from './mapapi/api_openlayers.js';
import PositionDialogModel from './positiondialogmodel.js';
import ViewModel from './viewmodel.js';
import uAlert from './alert.js';
import uDialog from './dialog.js';
import uObserve from './observe.js';
import uUtils from './utils.js';
/**
* @typedef {Object} MapViewModel.api
* @interface
* @memberOf MapViewModel
* @type {Object}
* @property {function(MapViewModel)} init
* @property {function} cleanup
* @property {function(uTrack, boolean)} displayTrack
* @property {function} clearMap
* @property {function(number)} animateMarker
* @property {function} getBounds
* @property {function} zoomToExtent
* @property {function} zoomToBounds
* @property {function} updateSize
* @property {function} updateState
*/
/**
* @typedef {Object} MapParams
* @property {number[]} center
* @property {number} zoom
* @property {number} rotation
*/
/**
* @class MapViewModel
*/
export default class MapViewModel extends ViewModel {
/**
* @param {uState} state
*/
constructor(state) {
super({
/** @type {?number} */
markerOver: null,
/** @type {?number} */
markerSelect: null,
// click handler
onMenuToggle: null
});
this.model.onMenuToggle = () => this.onMapResize();
this.state = state;
/** @type HTMLElement */
this.mapElement = document.querySelector('#map-canvas');
this.savedBounds = null;
this.api = null;
}
/**
* @return {MapViewModel}
*/
init() {
this.bindAll();
this.setObservers();
return this;
}
/**
* Dynamic change of map api
* @param {string} apiName API name
*/
loadMapAPI(apiName) {
let mapApi = this.api;
this.api = null;
if (mapApi) {
try {
this.savedBounds = mapApi.getBounds();
} catch (e) {
this.savedBounds = null;
}
mapApi.cleanup();
}
mapApi = this.getApi(apiName);
mapApi.init()
.then(() => {
this.api = mapApi;
this.onReady();
})
.catch((e) => {
let txt = $._('apifailure', apiName);
if (e && e.message) {
txt += ` (${e.message})`;
}
uAlert.error(txt, e);
config.mapApi = (apiName === 'gmaps') ? 'openlayers' : 'gmaps';
});
}
/**
* @param {string} apiName
* @return {OpenLayersApi|GoogleMapsApi}
*/
getApi(apiName) {
return apiName === 'gmaps' ? new GoogleMapsApi(this) : new OpenLayersApi(this);
}
onReady() {
if (this.state.currentTrack) {
let update = true;
if (this.savedBounds) {
this.api.zoomToBounds(this.savedBounds);
update = false;
}
this.displayTrack(this.state.currentTrack, update);
}
}
setObservers() {
config.onChanged('mapApi', (mapApi) => {
this.loadMapAPI(mapApi);
});
this.state.onChanged('currentTrack', (track) => {
if (!this.api) {
return;
}
this.api.clearMap();
if (track) {
uObserve.observe(track, 'positions', () => {
this.displayTrack(track, false);
this.api.zoomToExtent();
});
this.displayTrack(track, true);
}
});
this.state.onChanged('history', () => {
const history = this.state.history;
if (this.api && history && !history.trackId) {
if (history.mapApi) {
config.mapApi = history.mapApi;
} else {
if (history.mapParams) {
this.api.updateState(history.mapParams);
} else {
this.api.zoomToExtent();
}
this.state.history = null;
}
}
});
}
/**
* @param {uTrack} track Track to display
* @param {boolean} update Should update map view
*/
displayTrack(track, update) {
this.state.jobStart();
if (update && this.state.history && this.state.history.mapParams) {
this.api.updateState(this.state.history.mapParams);
update = false;
}
this.state.history = null;
this.api.displayTrack(track, update)
.finally(() => this.state.jobStop());
}
/**
* Get popup html
* @param {number} id Position index
* @returns {HTMLDivElement}
*/
getPopupElement(id) {
const pos = this.state.currentTrack.positions[id];
const count = this.state.currentTrack.length;
const user = this.state.currentTrack.user;
const isEditable = auth.user && (auth.isAdmin || auth.user.id === user.id);
let date = '–––';
let time = '–––';
if (pos.timestamp > 0) {
const dateTime = uUtils.getTimeString(new Date(pos.timestamp * 1000));
date = dateTime.date;
time = `${dateTime.time}${dateTime.zone}`;
}
let provider = '';
if (pos.provider === 'gps') {
provider = `
`;
} else if (pos.provider === 'network') {
provider = `
`;
}
let editLink = '';
if (isEditable) {
editLink = ``;
}
let stats = '';
if (!this.state.showLatest) {
stats =
`
data:image/s3,"s3://crabby-images/ce7de/ce7de25286049f3df5128c7b8a79793f5ae10e36" alt="${$._('ttime')} ${$._('ttime')}"
${$.getLocaleDuration(pos.totalSeconds)}
data:image/s3,"s3://crabby-images/4be8c/4be8c7e5b5d8a4eedfc15c0d380afc9931d278ff" alt="${$._('aspeed')} ${$._('aspeed')}"
${$.getLocaleSpeed(pos.totalSpeed, true)}
data:image/s3,"s3://crabby-images/cd520/cd520f17257e0055ae3eef6d9b4da8a8a9f86f0e" alt="${$._('tdistance')} ${$._('tdistance')}"
${$.getLocaleDistanceMajor(pos.totalMeters, true)}
`;
}
const html =
`
${(pos.hasComment()) ? `` : ''}
${(pos.hasImage()) ? `
` : ''}
data:image/s3,"s3://crabby-images/671f3/671f30d92e68662a351e92e8ef28875cb47240ea" alt="${$._('time')} ${$._('time')}"
${date}
data:image/s3,"s3://crabby-images/9851c/9851ccaa87dd91ec2088eddc22946ed1081d9c10" alt="${$._('time')} ${$._('time')}"
${time}
${(pos.speed !== null) ? `
data:image/s3,"s3://crabby-images/2d41c/2d41cd7c616ac061893119005c30ec1603fabf05" alt="${$._('speed')} ${$._('speed')}"
${$.getLocaleSpeed(pos.speed, true)}
` : ''}
${(pos.altitude !== null) ? `
data:image/s3,"s3://crabby-images/955a9/955a97f8d6e30624adecede6e1ee8771fc0a0589" alt="${$._('altitude')} ${$._('altitude')}"
${$.getLocaleAltitude(pos.altitude, true)}
` : ''}
${(pos.accuracy !== null) ? `
data:image/s3,"s3://crabby-images/1da61/1da610ab048779c3418375dcaf988adac6b05e7f" alt="${$._('accuracy')} ${$._('accuracy')}"
${$.getLocaleAccuracy(pos.accuracy, true)}${provider}
` : ''}
${(pos.bearing !== null) ? `
data:image/s3,"s3://crabby-images/4ec8d/4ec8ddda5c40720b56339f88fbc74a2c491bf28a" alt="${$._('bearing')} ${$._('bearing')}"
${pos.bearing}°
` : ''}
data:image/s3,"s3://crabby-images/815f7/815f78a63ff4fefc9258a79218a74cdc78da8919" alt="${$._('position')} ${$._('position')}"
${$.getLocaleCoordinates(pos)}
${stats}
`;
const node = document.createElement('div');
node.setAttribute('id', 'popup');
node.innerHTML = html;
if (pos.hasImage()) {
const image = node.querySelector('#pimage img');
image.onclick = () => {
const modal = new uDialog(`