- // ==UserScript==
- // @name WME E95
- // @version 0.4.12
- // @description Setup road properties in one click
- // @author Anton Shevchuk
- // @license MIT License
- // @include https://www.waze.com/editor*
- // @include https://www.waze.com/*/editor*
- // @include https://beta.waze.com/editor*
- // @include https://beta.waze.com/*/editor*
- // @exclude https://www.waze.com/user/editor*
- // @exclude https://beta.waze.com/user/editor*
- // @icon 
- // @grant none
- // @supportURL https://github.com/AntonShevchuk/wme-e95/issues
- // @namespace https://greasyfork.org/uk/scripts/382614-wme-e95
- // ==/UserScript==
- /* jshint esversion: 6 */
- /* global require, window */
-
- (function ($, WazeApi, I18n) {
- 'use strict';
-
- // Script name, uses as unique index
- const NAME = 'E95';
-
- // Translations
- const LOCALE = I18n.currentLocale();
- const translation = {
- 'en': {
- title: 'Quick Properties'
- },
- 'uk': {
- title: 'Швидкі налаштування',
- },
- 'ru': {
- title: 'Быстрые настройки'
- }
- };
- // Road Types
- // I18n.translations.uk.segment.road_types
- const types = {
- street: 1,
- primary: 2,
- // ...
- offroad: 8,
- // ...
- private: 17,
- // ...
- parking: 20,
- };
- // 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,
- // ??? : 0b00000010,
- // ??? : 0b00000100,
- // ??? : 0b00001000,
- unpaved: 0b00010000,
- headlights: 0b00100000,
- };
- // Buttons:
- // title - for buttons
- // keyCode - key for shortcuts (Alt + 1..9)
- // detectCity - try to detect city name by closures segments
- // clearCity - clear city name
- // attributes - native settings for model object
- // TODO:
- // – check permissions for user level lower than 2
- const buttons = {
- A: {
- title: 'PLR',
- shortcut: 'A+49',
- detectCity: true,
- attributes: {
- fwdMaxSpeed: 5,
- revMaxSpeed: 5,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: types.parking,
- flags: 0,
- lockRank: 0,
- }
- },
- B: {
- title: 'Pr20',
- shortcut: 'A+50',
- detectCity: true,
- attributes: {
- fwdMaxSpeed: 20,
- revMaxSpeed: 20,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: types.private,
- flags: 0,
- lockRank: 0,
- }
- },
- C: {
- title: 'Pr50',
- shortcut: 'A+51',
- detectCity: true,
- attributes: {
- fwdMaxSpeed: 50,
- revMaxSpeed: 50,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: types.private,
- flags: 0,
- lockRank: 0,
- }
- },
- D: {
- title: 'St50',
- shortcut: 'A+52',
- detectCity: true,
- attributes: {
- fwdMaxSpeed: 50,
- revMaxSpeed: 50,
- roadType: types.street,
- flags: 0,
- lockRank: 0,
- }
- },
- E: {
- title: 'PS50',
- shortcut: 'A+53',
- detectCity: true,
- attributes: {
- fwdMaxSpeed: 50,
- revMaxSpeed: 50,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: types.primary,
- flags: 0,
- lockRank: 1,
- }
- },
- F: {
- title: 'OR',
- shortcut: 'A+54',
- clearCity: true,
- attributes: {
- fwdMaxSpeed: 90,
- revMaxSpeed: 90,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: types.offroad,
- lockRank: 0,
- }
- },
- G: {
- title: 'Pr90',
- shortcut: 'A+55',
- clearCity: true,
- attributes: {
- fwdMaxSpeed: 90,
- revMaxSpeed: 90,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: types.private,
- lockRank: 0,
- }
- },
- H: {
- title: 'St90',
- shortcut: 'A+56',
- clearCity: true,
- attributes: {
- fwdMaxSpeed: 90,
- revMaxSpeed: 90,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: types.street,
- lockRank: 0,
- }
- },
- I: {
- title: 'PS90',
- shortcut: 'A+57',
- clearCity: true,
- attributes: {
- fwdMaxSpeed: 90,
- revMaxSpeed: 90,
- fwdMaxSpeedUnverified: false,
- revMaxSpeedUnverified: false,
- roadType: types.primary,
- lockRank: 1,
- }
- },
- };
- // Regions settings, will be merged with default values
- // Default values is actual for Ukraine
- const speed = {
- '20': {
- fwdMaxSpeed: 20,
- revMaxSpeed: 20,
- },
- '60': {
- fwdMaxSpeed: 60,
- revMaxSpeed: 60,
- }
- };
- const preset = {
- headlights: {
- attributes: {
- flags: flags.headlights
- }
- },
- pr60: {
- title: 'Pr60',
- attributes: speed["60"]
- },
- st60: {
- title: 'St60',
- attributes: speed["60"]
- },
- ps60: {
- title: 'PS60',
- attributes: speed["60"]
- },
- };
- const region = {
- // Belarus
- 'BO': {
- A: {
- attributes: speed["20"]
- },
- C: preset.pr60,
- D: preset.st60,
- E: preset.ps60,
- F: {
- title: 'SUP',
- attributes: {
- roadType: types.street,
- flags: flags.unpaved,
- }
- }
- },
- // Russian Federation
- 'RS': {
- C: preset.pr60,
- D: preset.st60,
- E: preset.ps60,
- },
- // Ukraine
- 'UP': {
- F: preset.headlights,
- G: preset.headlights,
- H: preset.headlights,
- I: preset.headlights,
- }
- };
-
- // Require Waze API
- let WazeActionUpdateObject = require('Waze/Action/UpdateObject');
- let WazeActionUpdateFeatureAddress = require('Waze/Action/UpdateFeatureAddress');
-
- // Get Button settings
- function getButtonConfig(index) {
- let btn = {};
- let abbr = WazeApi.model.getTopCountry().getAttributes().abbr;
- if (region[abbr] && region[abbr][index]) {
- // Merge default settings with region settings
- $.extend(true, btn, buttons[index], region[abbr][index]);
- } else {
- btn = buttons[index];
- }
- return btn;
- }
-
- // Update segment attributes
- function setupRoad(segment, settings, options = []) {
- let addr = segment.getAddress().attributes;
- // Change address
- let address = {
- countryID: addr.country ? addr.country.id : WazeApi.model.getTopCountry().getID(),
- stateID: addr.state ? addr.state.id : WazeApi.model.getTopState().getID(),
- cityName: addr.city ? addr.city.attributes.name : null,
- streetName: addr.street ? addr.street.name : null,
- };
- // Settings: Clear city
- if (settings.clearCity) {
- address.cityName = null;
- }
- // Settings: Detect city
- if (settings.detectCity && options.cityName) {
- address.cityName = options.cityName;
- }
- // Check city
- address.emptyCity = (address.cityName === null);
- // Check street
- address.emptyStreet = (address.streetName === null) || (address.streetName === '');
- // Update segment properties
- WazeApi.model.actionManager.add(
- new WazeActionUpdateObject(
- segment,
- settings.attributes
- )
- );
- // Update segment address
- WazeApi.model.actionManager.add(
- new WazeActionUpdateFeatureAddress(
- segment,
- address,
- {
- streetIDField: 'primaryStreetID'
- }
- )
- );
- }
-
- // Update street handler
- function processHandler() {
- process(this.dataset.e95);
- }
-
- function process(index) {
- // Get all selected segments
- let selected = WazeApi.selectionManager.getSelectedFeatures();
- let segments = [];
- let options = {};
- // Fill segments array
- for (let i = 0, total = selected.length; i < total; i++) {
- segments.push(WazeApi.model.segments.getObjectById(selected[i].model.attributes.id))
- }
- // Filter segments array
- segments = segments.filter(segment => segment && segment.getPermissions());
- // Try to detect city
- if (getButtonConfig(index).detectCity) {
- let cityName = null;
- for (let i = 0, total = segments.length; i < total; i++) {
- cityName = detectCity(segments[i]);
- if (cityName) {
- options.cityName = cityName;
- break;
- }
- }
- log('detected city ' + cityName);
- }
-
- for (let i = 0, total = segments.length; i < total; i++) {
- setupRoad(segments[i], getButtonConfig(index), options);
- }
- }
-
- // Detect city name by connected segments
- function detectCity(segment) {
- // Check cityName of the segment
- if (segment.getAddress().getCity() && !segment.getAddress().getCity().isEmpty()) {
- return segment.getAddress().getCity().getName();
- }
- let cityName = null;
- // TODO: replace follow magic with
- // segment.getConnectedSegments() and segment.getConnectedSegmentsByDirection() when it will work
- // last check - 30.07.19
- let connected = WazeApi.model.nodes.getObjectById(segment.getAttributes().fromNodeID).getSegmentIds(); // segments from point A
- connected = connected.concat(WazeApi.model.nodes.getObjectById(segment.getAttributes().toNodeID).getSegmentIds()); // segments from point B
- connected.filter(id => id !== segment.getID());
-
- for (let i = 0, total = connected.length; i < total; i++) {
- let city = WazeApi.model.segments.getObjectById(connected[i]).getAddress().getCity();
- // skip segments with empty cities
- if (city && !city.isEmpty()) {
- cityName = city.getName();
- break;
- }
- }
- return cityName;
- }
-
- // Create UI controls everytime when updated DOM of sidebar
- // Uses native JS function for better performance
- function createUI() {
- // Container for buttons
- let controls = document.createElement('div');
- controls.className = 'controls';
- // Create buttons
- for (let btn in buttons) {
- let config = getButtonConfig(btn);
- let button = document.createElement('button');
- button.className = 'waze-btn waze-btn-small e95 e95-' + btn;
- button.style.backgroundColor = colors[config.attributes.roadType];
- button.innerHTML = config.title;
- button.title = I18n.translate('segment.road_types')[config.attributes.roadType];
- button.dataset.e95 = btn;
- controls.appendChild(button);
- }
-
- let label = document.createElement('label');
- label.className = 'control-label';
- label.innerHTML = I18n.translate(NAME)['title'];
-
- let group = document.createElement('div');
- group.className = 'form-group ' + NAME;
- group.appendChild(label);
- group.appendChild(controls);
-
- document.getElementById('segment-edit-general').prepend(group);
- }
-
- // Apply CSS styles
- function appendStyle(css) {
- let style = document.createElement('style');
- style.type = 'text/css';
- style.innerHTML = css;
- document.getElementsByTagName('head')[0].appendChild(style);
- }
-
- // Simple console.log wrapper
- function log(message) {
- console.log(NAME + ': ' + message);
- }
-
- // Initial Translation for UI and Shortcuts
- function initTranslation() {
- I18n.translations[LOCALE][NAME] = translation[LOCALE] || translation['en'];
-
- // Translation for Shortcuts
- I18n.translations[LOCALE].keyboard_shortcuts.groups[NAME] = [];
- I18n.translations[LOCALE].keyboard_shortcuts.groups[NAME].description = NAME;
- I18n.translations[LOCALE].keyboard_shortcuts.groups[NAME].members = [];
-
- // Create description for every button
- for (let btn in buttons) {
- let name = NAME + 'Button' + buttons[btn].title;
- // Build description
- I18n.translations[LOCALE].keyboard_shortcuts.groups[NAME].members[name] =
- buttons[btn].title + ' - ' +
- I18n.translate('segment.road_types')[buttons[btn].attributes.roadType] + '; ' +
- I18n.translate('edit.segment.fields.speed_limit') + ' ' +
- I18n.translate('measurements.speed.km', {speed: buttons[btn].attributes.fwdMaxSpeed})
- ;
- }
- }
-
- // Initial Mutation Observer
- // #segment-edit-general - for segment tab
- // #landmark-edit-general - for POI tab
- function initObserver() {
- // Check for changes in the edit-panel
- // TODO: try to find solutions to handle native event
- let speedLimitsObserver = new MutationObserver(function (mutations) {
- mutations.forEach(function (mutation) {
- for (let i = 0, total = mutation.addedNodes.length; i < total; i++) {
- let node = mutation.addedNodes[i];
- // Only fire up if it's a node
- if (node.nodeType === Node.ELEMENT_NODE &&
- node.querySelector('div.selection') &&
- node.querySelector('#segment-edit-general') && // segment tab
- node.querySelector('div.hide-walking-trail').style.display !== 'none' && // skip for walking trails
- !node.querySelector('div.form-group.' + NAME)) {
- createUI();
- }
- }
- });
- });
-
- speedLimitsObserver.observe(document.getElementById('edit-panel'), {childList: true, subtree: true});
- log('observer was run');
- }
-
- function initButtons() {
- $('#edit-panel').on('click', 'button.e95', processHandler);
- }
-
- function initShortcuts() {
- WazeApi.accelerators.Groups[NAME] = [];
- WazeApi.accelerators.Groups[NAME].members = [];
-
- for (let btn in buttons) {
- let name = NAME + 'Button' + buttons[btn].title;
- WazeApi.accelerators.addAction(name, { group: NAME });
- WazeApi.accelerators.events.register(name, null, () => process(btn));
- WazeApi.accelerators.registerShortcut(buttons[btn].shortcut, name);
- }
- }
-
- function init() {
- // Initial Translation
- initTranslation();
-
- // Initial Mutation Observer
- initObserver();
-
- // Handler for all buttons
- initButtons();
-
- // Handler for button shortcuts
- initShortcuts();
-
- // Apply CSS styles
- appendStyle(
- '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-F { margin-right: 50px; }'
- );
- }
-
- // Bootstrap plugin
- function bootstrap(tries = 1) {
- log('attempt ' + tries);
- if (WazeApi &&
- WazeApi.map &&
- WazeApi.model &&
- WazeApi.loginManager.user) {
- log('was initialized');
- init();
- } else if (tries < 100) {
- tries++;
- setTimeout(() => bootstrap(tries), 500);
- } else {
- console.error('initialization failed');
- }
- }
-
- log('initialization');
- bootstrap();
- })(window.jQuery, window.W, window.I18n);