- // ==UserScript==
- // @name WME E95
- // @version 0.7.0
- // @description Setup road properties in one click
- // @license MIT License
- // @author Anton Shevchuk
- // @namespace https://greasyfork.org/users/227648-anton-shevchuk
- // @supportURL https://github.com/AntonShevchuk/wme-e95/issues
- // @match https://*.waze.com/editor*
- // @match https://*.waze.com/*/editor*
- // @exclude https://*.waze.com/user/editor*
- // @icon 
- // @grant none
- // @require https://greasyfork.org/scripts/389765-common-utils/code/CommonUtils.js?version=1090053
- // @require https://greasyfork.org/scripts/450160-wme-bootstrap/code/WME-Bootstrap.js?version=1135567
- // @require https://greasyfork.org/scripts/452563-wme/code/WME.js?version=1101598
- // @require https://greasyfork.org/scripts/450221-wme-base/code/WME-Base.js?version=1137043
- // @require https://greasyfork.org/scripts/450320-wme-ui/code/WME-UI.js?version=1137289
- // ==/UserScript==
-
- /* jshint esversion: 8 */
- /* global require */
- /* global $, jQuery */
- /* global W */
- /* global I18n */
- /* global WME, WMEBase, WMEUI, WMEUIHelper, WMEUIShortcut, WMEUIHelperControlButton */
- /* global Container, Settings, SimpleCache, Tools */
-
- (function () {
- 'use strict'
-
- // Script name, uses as unique index
- const NAME = 'E95'
-
- // Translations
- const TRANSLATION = {
- 'en': {
- title: 'Quick Properties',
- },
- 'uk': {
- title: 'Швидкі налаштування',
- },
- 'ru': {
- title: 'Быстрые настройки'
- }
- }
-
- const STYLE = 'button.waze-btn.E95 { margin: 0 4px 4px 0; padding: 2px; width: 42px; } ' +
- 'button.waze-btn.E95:hover { box-shadow: 0 2px 8px 0 rgba(0, 0, 0, 0.1), inset 0 0 100px 100px rgba(255, 255, 255, 0.3); } ' +
- 'button.waze-btn.E95-E { margin-right: 42px; }' +
- 'button.waze-btn.E95-J { margin-right: 42px; }'
-
- WMEUI.addTranslation(NAME, TRANSLATION)
- WMEUI.addStyle(STYLE)
-
- // Road Types
- // I18n.translations.uk.segment.road_types
- // I18n.translations.en.segment.road_types
- const TYPES = {
- street: 1,
- primary: 2,
- freeway: 3,
- ramp: 4,
- trail: 5,
- major: 6,
- minor: 7,
- offroad: 8,
- walkway: 9,
- boardwalk: 10,
- ferry: 15,
- stairway: 16,
- private: 17,
- railroad: 18,
- runway: 19,
- parking: 20,
- narrow: 22
- }
- // Road colors by type
- const COLORS = {
- '1': '#ffffeb',
- '2': '#f0ea58',
- // ...
- '8': '#867342',
- // ...
- '17': '#beba6c',
- // ...
- '20': '#ababab'
- }
- // Road Flags
- // for setup flags use binary operators
- // e.g. flags.tunnel | flags.headlights
- const FLAGS = {
- tunnel: 0b00000001,
- // ???
- // a : 0b00000010,
- // b : 0b00000100,
- // c : 0b00001000,
- unpaved: 0b00010000,
- headlights: 0b00100000,
- }
- // Buttons:
- // title - for buttons
- // shortcut - keys for shortcuts, by default is Alt + (1..9)
- // options:
- // - detectCity - try to detect the city name by closures segments
- // - clearCity - clear the city name
- // - clearStreet - clear the street name
- // attributes - native settings for model object
- // TODO:
- // – check permissions for user level lower than 2
- const BUTTONS = {
- A: {
- title: 'Pr 5',
- shortcut: 'A+1',
- options: {
- detectCity: true,
- },
- attributes: {
- flags: 0,
- fwdMaxSpeed: 5,
- revMaxSpeed: 5,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: TYPES.private,
- lockRank: 0,
- }
- },
- B: {
- title: 'Pr20',
- shortcut: 'A+2',
- options: {
- detectCity: true,
- },
- attributes: {
- flags: 0,
- fwdMaxSpeed: 20,
- revMaxSpeed: 20,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: TYPES.private,
- lockRank: 0,
- }
- },
- C: {
- title: 'Pr50',
- shortcut: 'A+3',
- options: {
- detectCity: true,
- },
- attributes: {
- flags: 0,
- fwdMaxSpeed: 50,
- revMaxSpeed: 50,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: TYPES.private,
- lockRank: 0,
- }
- },
- D: {
- title: 'St50',
- shortcut: 'A+4',
- options: {
- detectCity: true,
- },
- attributes: {
- flags: 0,
- fwdMaxSpeed: 50,
- revMaxSpeed: 50,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: TYPES.street,
- lockRank: 0,
- }
- },
- E: {
- title: 'PS50',
- shortcut: 'A+5',
- options: {
- detectCity: true,
- },
- attributes: {
- flags: 0,
- fwdMaxSpeed: 50,
- revMaxSpeed: 50,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: TYPES.primary,
- lockRank: 1,
- }
- },
- F: {
- title: 'PLR',
- shortcut: 'A+6',
- options: {
- detectCity: true,
- },
- attributes: {
- flags: 0,
- fwdMaxSpeed: 5,
- revMaxSpeed: 5,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: TYPES.parking,
- lockRank: 0,
- }
- },
- G: {
- title: 'OR',
- shortcut: 'A+7',
- options: {
- clearCity: true,
- clearStreet: false,
- },
- attributes: {
- flags: 0,
- fwdMaxSpeed: 90,
- revMaxSpeed: 90,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: TYPES.offroad,
- lockRank: 0,
- }
- },
- H: {
- title: 'Pr90',
- shortcut: 'A+8',
- options: {
- clearCity: true,
- },
- attributes: {
- flags: 0,
- fwdMaxSpeed: 90,
- revMaxSpeed: 90,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: TYPES.private,
- lockRank: 0,
- }
- },
- I: {
- title: 'St90',
- shortcut: 'A+9',
- options: {
- clearCity: true,
- },
- attributes: {
- flags: 0,
- fwdMaxSpeed: 90,
- revMaxSpeed: 90,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: TYPES.street,
- lockRank: 0,
- }
- },
- J: {
- title: 'PS90',
- shortcut: 'A+0',
- options: {
- clearCity: true,
- },
- attributes: {
- flags: 0,
- fwdMaxSpeed: 90,
- revMaxSpeed: 90,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: TYPES.primary,
- lockRank: 1,
- }
- }
- }
-
- // country specified buttons config
- const CONFIGS = {
- // Ukraine
- 232: {
- G: {
- attributes: {
- flags: FLAGS.headlights
- }
- },
- H: {
- attributes: {
- flags: FLAGS.headlights
- }
- },
- I: {
- attributes: {
- flags: FLAGS.headlights
- }
- },
- J: {
- attributes: {
- flags: FLAGS.headlights
- }
- },
- }
- }
-
- // Require Waze API
- let WazeActionUpdateObject
- let WazeActionUpdateFeatureAddress
-
- class E95 extends WMEBase {
- constructor (name, buttons, configs) {
- super(name)
-
- this.panel = null
- this.buttons = buttons
- this.configs = configs
- }
-
- getPanel () {
- if (this.panel) {
- return this.panel
- }
-
- // Build panel
- // Container for buttons
- let controls = document.createElement('div')
- controls.className = 'controls'
- // Create buttons
- for (let btn in this.buttons) {
- let config = this.getButtonConfig(btn)
- let title = config.title
- let color = COLORS[config.attributes.roadType]
- let description = config.title + ' - ' +
- I18n.t('segment.road_types')[config.attributes.roadType] + '; ' +
- I18n.t('edit.segment.fields.speed_limit') + ' ' +
- I18n.t('measurements.speed.km', { speed: config.attributes.fwdMaxSpeed })
-
- let UIButton = new WMEUIHelperControlButton(
- NAME,
- btn,
- title,
- description,
- () => this.setupCallback(config),
- config.shortcut
- )
- let button = UIButton.html()
- button.dataset[NAME] = btn
- button.style.backgroundColor = color
- controls.appendChild(button)
- }
- let label = document.createElement('label')
- label.className = 'control-label'
- label.innerText = I18n.t(NAME).title
-
- this.panel = document.createElement('div')
- this.panel.className = 'form-group ' + NAME
- this.panel.appendChild(label)
- this.panel.appendChild(controls)
-
- return this.panel
- }
-
- // Get Button settings
- getButtonConfig (index) {
- // Load settings for current country by call method W.model.getTopCountry().getID()
- // Then mixed it with default settings by Tools.mergeDeep() method
- return Tools.mergeDeep(this.buttons[index], this.configs[W.model.getTopCountry().getID()][index])
- }
-
- // Handler for Road buttons
- setupCallback (button) {
- // Get all selected segments
- let segments = WME.getSelectedSegments()
-
- // Try to detect city, if needed
- if (button.options.detectCity) {
- let cityName = null
- for (let i = 0, total = segments.length; i < total; i++) {
- cityName = this.detectCity(segments[i])
- if (cityName) {
- button.options.cityName = cityName
- break
- }
- }
- }
- for (let i = 0, total = segments.length; i < total; i++) {
- this.updateSegment(segments[i], button.options, button.attributes)
- }
- }
-
- /**
- * Update segment attributes
- * @param {Object} segment
- * @param {Object} options
- * @param {Object} attributes
- */
- updateSegment (segment, options, attributes = {}) {
- // current segment address
- let addr = segment.getAddress().attributes
- // fill address information
- let address = {
- countryID: addr.country ? addr.country.id : W.model.getTopCountry().getID(),
- stateID: addr.state ? addr.state.id : W.model.getTopState().getID(),
- cityName: addr.city ? addr.city.attributes.name : null,
- streetName: addr.street ? addr.street.name : null,
- }
- // options: detect city
- if (!address.cityName && options.detectCity && options.cityName) {
- this.log('detected city name "' + options.cityName + '"')
- address.cityName = options.cityName
- }
- // options: clear city
- if (options.clearCity) {
- this.log('clear city name')
- address.cityName = null
- }
- // options: clear street
- if (options.clearStreet) {
- this.log('clear street name')
- address.streetName = null
- }
- // set city flag
- address.emptyCity = (address.cityName === null)
- // set street flag
- address.emptyStreet = (address.streetName === null) || (address.streetName === '')
- // update segment's address
- W.model.actionManager.add(
- new WazeActionUpdateFeatureAddress(
- segment,
- address,
- {
- streetIDField: 'primaryStreetID'
- }
- )
- )
- // keep the current lock level if it is higher than in the config's attributes
- if (segment.attributes.lockRank > attributes.lockRank) {
- attributes.lockRank = segment.attributes.lockRank
- }
- // need more logs
- this.log('set road type to ' + I18n.t('segment.road_types')[attributes.roadType])
- // update segment's properties
- W.model.actionManager.add(
- new WazeActionUpdateObject(
- segment,
- attributes
- )
- )
- }
-
- /**
- * Detect city name by connected segments
- * @param {Object} segment
- * @return {String|null}
- */
- detectCity (segment) {
- // check cityName of the segment
- if (segment.getAddress().getCity() && !segment.getAddress().getCity().isEmpty()) {
- return segment.getAddress().getCity().getName()
- }
-
- // TODO: replace follow magic with methods getConnectedSegments() and getConnectedSegmentsByDirection()
- // W.selectionManager.getSelectedFeatures()[0].model.getConnectedSegments().map(x => x.attributes.id)
- // W.selectionManager.getSelectedFeatures()[0].model.getConnectedSegmentsByDirection().map(x => x.attributes.id)
- // when it will work
- // last check - 2022.12.29
-
- // connected segments
- let connected = []
- connected = connected.concat(segment.getFromNode().getSegmentIds()) // segments from point A
- connected = connected.concat(segment.getToNode().getSegmentIds()) // segments from point B
- connected = connected.filter(id => id !== segment.getID()) // filter himself
-
- // cities of the connected segments
- let cities = connected.map(id => W.model.segments.getObjectById(id).getAddress().getCity())
- cities = cities.filter(city => city) // filter segments w/out city
- cities = cities.map(city => city.getName()) // extract cities name
- cities = cities.filter(city => city) // filter empty city name
-
- if (cities.length) {
- return cities.shift()
- }
- return null
- }
-
- /**
- * Handler for `segment.wme` event
- * Create UI controls every time when updated DOM of sidebar
- * Uses native JS function for better performance
- *
- * @param {jQuery.Event} event
- * @param {HTMLElement} element
- * @param {W.model} model
- * @return {void}
- */
- onSegment (event, element, model) {
- // Skip for walking trails
- if (model.isWalkingRoadType()) {
- return
- }
- // Panel should already create
- element.prepend(this.getPanel())
- }
-
- /**
- * Handler for `segments.wme` event
- * Create UI controls every time when updated DOM of sidebar
- * Uses native JS function for better performance
- *
- * @param {jQuery.Event} event
- * @param {HTMLElement} element
- * @param {Array} models
- * @return {void}
- */
- onSegments (event, element, models) {
- // Skip for walking trails
- if (models.filter((model) => !model.isWalkingRoadType()).length === 0) {
- return
- }
- // Panel should already create
- element.prepend(this.getPanel())
- }
- }
-
- $(document).on('bootstrap.wme', () => {
- // Require scripts
- WazeActionUpdateObject = require('Waze/Action/UpdateObject')
- WazeActionUpdateFeatureAddress = require('Waze/Action/UpdateFeatureAddress')
-
- new E95(NAME, BUTTONS, CONFIGS)
-
- WMEUIShortcut.setGroupTitle(NAME, I18n.t(NAME).title)
- })
- })()