Create small previews for chosen map providers
// ==UserScript== // @name WME E58 Map's previews // @name:uk WME 🇺🇦 E58 Map's previews // @version 0.6.3 // @description Create small previews for chosen map providers // @description:uk Створює невеличку карту для перегляду // @license MIT License // @author Anton Shevchuk // @namespace https://greasyfork.org/users/227648-anton-shevchuk // @supportURL https://github.com/AntonShevchuk/wme-e58/issues // @match https://*.waze.com/editor* // @match https://*.waze.com/*/editor* // @exclude https://*.waze.com/user/editor* // @icon  // @grant none // @require https://update.greasyfork.org/scripts/389765/1090053/CommonUtils.js // @require https://update.greasyfork.org/scripts/450160/1218867/WME-Bootstrap.js // @require https://update.greasyfork.org/scripts/450221/1137043/WME-Base.js // @require https://update.greasyfork.org/scripts/450320/1555446/WME-UI.js // ==/UserScript== /* jshint esversion: 8 */ /* global require */ /* global $, jQuery */ /* global W */ /* global I18n */ /* global WME, WMEBase, WMEUI, WMEUIHelper */ /* global Container, Settings, SimpleCache, Tools */ /* global H, google */ (function () { 'use strict' const NAME = 'E58' // Translation const TRANSLATION = { 'en': { // Tab title title: 'Map preview', // Tab description description: 'Open a small preview modal window with the map', // Tab help help: 'You can use the <a href="#keyboard-dialog" target="_blank" rel="noopener noreferrer" data-toggle="modal">Keyboard shortcuts</a> to open the map preview window. It\'s more convenient than clicking on the button.', maps: { // Fieldset's legend title: 'Sources', // Fieldset's description description: 'Choose a map provider', // Description for option `Google` google: 'Google', // Description for option `OSM` osm: 'Open Street Map', }, options: { title: 'Options', description: 'Choose a map provider in the settings', controls: 'Controls on the map', interactive: 'Interaction with the map', }, }, 'uk': { title: 'Карта', description: 'Відкрити маленьку карту', help: 'Використовуйте <a href="#keyboard-dialog" target="_blank" rel="noopener noreferrer" data-toggle="modal">гарячі клавіши</a>, це значно швидше ніж використовувати кнопку', maps: { title: 'Джерела', description: 'Оберіть карту для відображення', google: 'Google', osm: 'Open Street Map', }, options: { title: 'Налаштування', description: 'Оберіть у налаштуваннях карту для відображення', controls: 'Елементи управління', interactive: 'Можливість взаємодіяти с картою', }, }, 'ru': { title: 'Карта', description: 'Открыть маленькую карту', help: 'Используйте <a href="#keyboard-dialog" target="_blank" rel="noopener noreferrer" data-toggle="modal">комбинации клавиш</a>, и не надо будет клацать кнопку', maps: { title: 'Источники', description: 'Выберите карту для отображения', google: 'Google', osm: 'Open Street Map', }, options: { title: 'Настройки', description: 'Выберите в настройках карту для отображения', controls: 'Элементи управления карты', interactive: 'Возможность взаимодествия с картой', }, } } const STYLE = '.e58 .header h5 { padding: 16px 16px 0; font-size: 16px }' + '.e58 legend { cursor:pointer; font-size: 12px; font-weight: bold; width: auto; text-align: right; border: 0; margin: 0; padding: 0 8px; }' + '.e58 fieldset { border: 1px solid #ddd; padding: 4px; }' + '.e58 fieldset p { padding: 0; margin: 0 8px !important; }' + '.e58 fieldset.e58 div.controls label { white-space: normal; font-weight: 400; }' + 'div.e58.e58-text { margin: 15px 0; }' + 'p.e58-info { border-top: 1px solid #ccc; color: #777; font-size: x-small; margin-top: 15px; padding-top: 10px; text-align: center; }' WMEUI.addTranslation(NAME, TRANSLATION) WMEUI.addStyle(STYLE) // Default settings const SETTINGS = { map: 'google', maps: [ 'google', 'osm' ], options: { controls: false, interactive: false, } } /** * Basic Map class */ class MapPreview { constructor (uid, container, settings) { this.uid = uid this.map = null this.wrapper = this._wrapper() container.append(this.wrapper) container.style.height = '256px' this.settings = settings this.controls = settings.get('options', 'controls') this.interactive = settings.get('options', 'interactive') } /** * Load external JS Map library * @param {String} url * @return {Promise<*>} */ async script (url) { return $.ajax({ url: url, cache: true, dataType: 'script', success: () => console.log(NAME, this.uid, 'loaded') }) } /** * Build div for map * @return {HTMLDivElement} * @protected */ _wrapper () { let div = document.createElement('div') div.id = this._uid() div.style.height = '256px' return div } _uid () { return NAME + '-map-' + this.uid } _center () { let center = new OpenLayers.Geometry.Point(W.map.getCenter().lon, W.map.getCenter().lat).transform('EPSG:900913', 'EPSG:4326') return { lon: center.x, lat: center.y, } } _zoom () { return W.map.getZoom() - 1 } update () { let center = this._center() this._update(center.lat, center.lon, this._zoom()) } _update (lat, lon, zoom) { throw new Error('Abstract method') } } /** * Google Maps */ class GooglePreview extends MapPreview { constructor (container, settings) { super('Google', container, settings) } async render () { let pos = this._center() this.map = new google.maps.Map(this.wrapper, { center: new google.maps.LatLng(pos.lat, pos.lon), zoom: this._zoom(), mapTypeId: 'roadmap', mapTypeControl: false, streetViewControl: false, disableDefaultUI: !this.controls, gestureHandling: this.interactive ? 'cooperative ' : 'none', zoomControl: this.controls, }) // Setup handler W.map.events.register('moveend', null, () => this.update()) } _update (lat, lon, zoom) { this.map.setZoom(zoom) this.map.setCenter(new google.maps.LatLng(lat, lon)) } } /** * Open Street Maps */ class OSMPreview extends MapPreview { constructor (container, settings) { super('OSM', container, settings) } async render () { let pos = this._center() this.map = new google.maps.Map(this.wrapper, { center: new google.maps.LatLng(pos.lat, pos.lon), zoom: this._zoom(), mapTypeId: 'OSM', mapTypeControl: false, streetViewControl: false, disableDefaultUI: !this.controls, gestureHandling: this.interactive ? 'cooperative ' : 'none', zoomControl: this.controls, }) // Define OSM map type pointing at the OpenStreetMap tile server this.map.mapTypes.set('OSM', new google.maps.ImageMapType({ getTileUrl: function (coord, zoom) { return 'https://tile.openstreetmap.org/' + zoom + '/' + coord.x + '/' + coord.y + '.png' }, tileSize: new google.maps.Size(256, 256), name: 'OpenStreetMap', maxZoom: 18 })) // Setup handler W.map.events.register('moveend', null, () => this.update()) } _update (lat, lon, zoom) { this.map.setZoom(zoom) this.map.setCenter(new google.maps.LatLng(lat, lon)) } } /** * E58 Map Preview class */ class E58 extends WMEBase { constructor (name, settings) { super(name, settings) this.helper = new WMEUIHelper(name) let tab = this.helper.createTab( I18n.t(name).title, { image: GM_info.script.icon } ) tab.addText('description', I18n.t(name).description) let button = tab.addButton('preview', I18n.t(name).title, '', () => this.toggleMap()) button.html().className += ' waze-btn-blue' // Setup providers map settings let fsMap = this.helper.createFieldset(I18n.t(name).maps.title) for (let i = 0; i < settings.maps.length; i++) { let map = settings.maps[i] fsMap.addRadio( 'maps-' + map, I18n.t(name).maps[map], () => this.settings.set(['map'], map), 'maps', map, this.settings.get('map') === map ) } tab.addElement(fsMap) // Setup options for maps let fsOptions = this.helper.createFieldset(I18n.t(name).options.title) for (let item in settings.options) { if (settings.options.hasOwnProperty(item)) { fsOptions.addCheckbox( 'options-' + item, I18n.t(name).options[item], (event) => this.settings.set(['options', item], event.target.checked), this.settings.get('options', item)) } } tab.addElement(fsOptions) tab.addDiv('text', I18n.t(name).help) tab.addText( 'info', '<a href="' + GM_info.scriptUpdateURL + '">' + GM_info.script.name + '</a> ' + GM_info.script.version ) tab.inject() } /** * Show modal with map preview */ toggleMap () { if (document.getElementById('e58-map-preview')) { this.log('hide preview map') $('#panel-container a.close-panel').click() return } /** @type {WMEUIHelperModal} */ let modal = this.helper.createModal( I18n.t(this.name).title ) // Setup Preview Map element let map = modal.addDiv('map-preview').html() modal.inject() this.log('show preview map', this.settings.get('map')) if (this.settings.get('map') === 'google') { let Google = new GooglePreview(map, this.settings) Google.render() } else if (this.settings.get('map') === 'osm') { let OSM = new OSMPreview(map, this.settings) OSM.render() } else { // disabled map.innerText = I18n.t(this.name).maps.description } } } // Main handler $(document).on('bootstrap.wme', () => { // Create E58 Instance let Instance = new E58(NAME, SETTINGS) // Bind shortcut WMEUI.addShortcut( NAME, I18n.t(NAME).description, NAME, I18n.t(NAME).title + ' 🗺️', 'A+M', () => Instance.toggleMap(), ) }) })()