WME E58 Map's previews

Create small previews for chosen map providers

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         WME E58 Map's previews
// @name:uk      WME 🇺🇦 E58 Map's previews
// @name:ru      WME 🇺🇦 E58 Map's previews
// @version      0.8.1
// @description  Create small previews for chosen map providers
// @description:uk Створює невеличку карту для перегляду
// @description:ru Создаёт небольшую карту для просмотра
// @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/1704233/WME-Bootstrap.js
// @require      https://update.greasyfork.org/scripts/450221/1691071/WME-Base.js
// @require      https://update.greasyfork.org/scripts/450320/1688694/WME-UI.js
// ==/UserScript==

/* jshint esversion: 8 */
/* global require */
/* global $, jQuery */
/* global I18n */
/* global WME, WMEBase, WMEUI, WMEUIHelper */
/* global Container, Settings, SimpleCache, Tools  */
/* global H, google */
/* global Node$1, Segment, Venue, VenueAddress, WmeSDK */

(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 <strong>Keyboard shortcuts</strong> 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: 'Використовуйте <strong>гарячі клавіши</strong>, це значно швидше ніж використовувати кнопку',
      maps: {
        title: 'Джерела',
        description: 'Оберіть карту для відображення',
        google: 'Google',
        osm: 'Open Street Map',
      },
      options: {
        title: 'Налаштування',
        description: 'Оберіть у налаштуваннях карту для відображення',
        controls: 'Елементи управління',
        interactive: 'Можливість взаємодіяти с картою',
      },
    },
    'ru': {
      title: 'Карта',
      description: 'Открыть маленькую карту',
      help: 'Используйте <strong>комбинации клавиш</strong>, и не надо будет клацать кнопку',
      maps: {
        title: 'Источники',
        description: 'Выберите карту для отображения',
        google: 'Google',
        osm: 'Open Street Map',
      },
      options: {
        title: 'Настройки',
        description: 'Выберите в настройках карту для отображения',
        controls: 'Элементы управления карты',
        interactive: 'Возможность взаимодействия с картой',
      },
    }
  }

  WMEUI.addTranslation(NAME, TRANSLATION)

  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; }' +
    '#sidebar p.e58-blue { background-color:#0057B8;color:white;height:32px;text-align:center;line-height:32px;font-size:24px;margin:0; }' +
    '#sidebar p.e58-yellow { background-color:#FFDD00;color:black;height:32px;text-align:center;line-height:32px;font-size:24px;margin:0; }'

  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.initHelper()

      this.initTab(settings)

      this.initShortcuts()
    }

    initHelper() {
      this.helper = new WMEUIHelper(this.name)
    }

    initTab (settings) {
      let tab = this.helper.createTab(
        I18n.t(this.name).title,
        {
          sidebar: this.wmeSDK.Sidebar,
          image: GM_info.script.icon
        }
      )
      tab.addText('description', I18n.t(this.name).description)
      let button = tab.addButton('preview', I18n.t(this.name).title, '', () => this.toggleMap())
      button.html().className += ' waze-btn-blue'

      // Setup providers map settings
      let fsMap = this.helper.createFieldset(I18n.t(this.name).maps.title)

      for (let i = 0; i < settings.maps.length; i++) {
        let map = settings.maps[i]
        fsMap.addRadio(
          'maps-' + map,
          I18n.t(this.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(this.name).options.title)
      for (let item in settings.options) {
        if (settings.options.hasOwnProperty(item)) {
          fsOptions.addCheckbox(
            'options-' + item,
            I18n.t(this.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(this.name).help)
      tab.addText(
        'info',
        '<a href="' + GM_info.scriptUpdateURL + '">' + GM_info.script.name + '</a> ' + GM_info.script.version
      )
      tab.addText('blue', 'made in')
      tab.addText('yellow', 'Ukraine')
      tab.inject()
    }

    initShortcuts () {
      let shortcut = {
        callback: () => this.toggleMap(),
        description: I18n.t(this.name).description,
        shortcutId: this.id,
        shortcutKeys: 'A+N',
      };

      if (this.wmeSDK.Shortcuts.areShortcutKeysInUse({ shortcutKeys: shortcut.shortcutKeys })) {
        this.log('Shortcut already in use')
        shortcut.shortcutKeys = null
      }
      this.wmeSDK.Shortcuts.createShortcut(shortcut);
    }

    /**
     * Show modal with map preview
     */
    toggleMap () {
      if (document.getElementById('e58-map-preview')) {
        this.log('hide preview map')
        $('.wme-ui-panel.e58 button.wme-ui-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', () => {
    new E58(NAME, SETTINGS)
  })
})()