- // ==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://gf.qytechs.cn/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.gf.qytechs.cn/scripts/389765/1090053/CommonUtils.js
- // @require https://update.gf.qytechs.cn/scripts/450160/1218867/WME-Bootstrap.js
- // @require https://update.gf.qytechs.cn/scripts/450221/1137043/WME-Base.js
- // @require https://update.gf.qytechs.cn/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(),
- )
- })
- })()