WME E95

Setup road properties in one click

当前为 2019-08-28 提交的版本,查看 最新版本

// ==UserScript==
// @name         WME E95
// @version      0.4.16
// @description  Setup road properties in one click
// @author       Anton Shevchuk
// @license      MIT License
// @grant        none
// @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         
// @supportURL   https://github.com/AntonShevchuk/wme-e95/issues
// @namespace    https://gf.qytechs.cn/users/227648
// ==/UserScript==
/* jshint esversion: 6 */
/* global require, window, W, I18n */

(function ($) {
    '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
    //   shortcut - keys for shortcuts, by default is 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',
            callback: process,
            detectCity: true,
            attributes: {
                fwdMaxSpeed: 5,
                revMaxSpeed: 5,
                fwdMaxSpeedUnverified: false,
                revMaxSpeedUnverified: false,
                roadType: types.parking,
                flags: 0,
                lockRank: 0,
            }
        },
        B: {
            title: 'Pr20',
            shortcut: 'A+50',
            callback: process,
            detectCity: true,
            attributes: {
                fwdMaxSpeed: 20,
                revMaxSpeed: 20,
                fwdMaxSpeedUnverified: false,
                revMaxSpeedUnverified: false,
                roadType: types.private,
                flags: 0,
                lockRank: 0,
            }
        },
        C: {
            title: 'Pr50',
            shortcut: 'A+51',
            callback: process,
            detectCity: true,
            attributes: {
                fwdMaxSpeed: 50,
                revMaxSpeed: 50,
                fwdMaxSpeedUnverified: false,
                revMaxSpeedUnverified: false,
                roadType: types.private,
                flags: 0,
                lockRank: 0,
            }
        },
        D: {
            title: 'St50',
            shortcut: 'A+52',
            callback: process,
            detectCity: true,
            attributes: {
                fwdMaxSpeed: 50,
                revMaxSpeed: 50,
                roadType: types.street,
                flags: 0,
                lockRank: 0,
            }
        },
        E: {
            title: 'PS50',
            shortcut: 'A+53',
            callback: process,
            detectCity: true,
            attributes: {
                fwdMaxSpeed: 50,
                revMaxSpeed: 50,
                fwdMaxSpeedUnverified: false,
                revMaxSpeedUnverified: false,
                roadType: types.primary,
                flags: 0,
                lockRank: 1,
            }
        },
        F: {
            title: 'OR',
            shortcut: 'A+54',
            callback: process,
            clearCity: true,
            attributes: {
                fwdMaxSpeed: 90,
                revMaxSpeed: 90,
                fwdMaxSpeedUnverified: false,
                revMaxSpeedUnverified: false,
                roadType: types.offroad,
                lockRank: 0,
            }
        },
        G: {
            title: 'Pr90',
            shortcut: 'A+55',
            callback: process,
            clearCity: true,
            attributes: {
                fwdMaxSpeed: 90,
                revMaxSpeed: 90,
                fwdMaxSpeedUnverified: false,
                revMaxSpeedUnverified: false,
                roadType: types.private,
                lockRank: 0,
            }
        },
        H: {
            title: 'St90',
            shortcut: 'A+56',
            callback: process,
            clearCity: true,
            attributes: {
                fwdMaxSpeed: 90,
                revMaxSpeed: 90,
                fwdMaxSpeedUnverified: false,
                revMaxSpeedUnverified: false,
                roadType: types.street,
                lockRank: 0,
            }
        },
        I: {
            title: 'PS90',
            shortcut: 'A+57',
            callback: process,
            clearCity: true,
            attributes: {
                fwdMaxSpeed: 90,
                revMaxSpeed: 90,
                fwdMaxSpeedUnverified: false,
                revMaxSpeedUnverified: false,
                roadType: types.primary,
                lockRank: 1,
            }
        },
        J: {
            title: 'U',
            shortcut: 'A+U',
            callback: function (button) {
                console.log(button);
                console.log(W.selectionManager.getSegmentSelection());
            }
        }
    };
    // 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
        37: {
            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
        186: {
            C: preset.pr60,
            D: preset.st60,
            E: preset.ps60,
        },
        // Ukraine
        232: {
            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 = W.model.getTopCountry().getID();
        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 : 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,
        };
        // 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
        W.model.actionManager.add(
          new WazeActionUpdateObject(
            segment,
            settings.attributes
          )
        );
        // Update segment address
        W.model.actionManager.add(
          new WazeActionUpdateFeatureAddress(
            segment,
            address,
            {
                streetIDField: 'primaryStreetID'
            }
          )
        );
    }

    // Update handler
    function processHandler() {
        let button = getButtonConfig(this.dataset[NAME]);
        // run callback
        return button.callback(button);
    }

    function process(button) {
        // Get all selected segments
        let selected = W.selectionManager.getSelectedFeatures();
        let segments = [];
        let options = {};
        // Fill segments array
        for (let i = 0, total = selected.length; i < total; i++) {
            segments.push(W.model.segments.getObjectById(selected[i].model.attributes.id))
        }
        // Filter segments array
        segments = segments.filter(segment => segment && segment.getPermissions());
        // Try to detect city
        if (button.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], button, 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 = W.model.nodes.getObjectById(segment.getAttributes().fromNodeID).getSegmentIds(); // segments from point A
        connected = connected.concat(W.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 = W.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.dataset[NAME] = btn;
                button.className = 'waze-btn waze-btn-small ' + NAME + ' ' + NAME + '-' + btn;
                button.innerHTML = config.title;
                if (config.attributes) {
                    button.title = I18n.t('segment.road_types')[config.attributes.roadType];
                    button.style.backgroundColor = colors[config.attributes.roadType];
                } else {
                    button.title = config.title;
                }

            controls.appendChild(button);
        }

        let label = document.createElement('label');
            label.className = 'control-label';
            label.innerHTML = I18n.t(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;
            if (buttons[btn].attributes) {
                I18n.translations[LOCALE].keyboard_shortcuts.groups[NAME].members[name] += ' - ' +
                I18n.t('segment.road_types')[buttons[btn].attributes.roadType] + '; ' +
                I18n.t('edit.segment.fields.speed_limit') + ' ' +
                I18n.t('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.'+NAME, processHandler);
    }

    function initShortcuts() {
        W.accelerators.Groups[NAME] = [];
        W.accelerators.Groups[NAME].members = [];

        for (let btn in buttons) {
            let name = NAME + 'Button' + buttons[btn].title;
            W.accelerators.addAction(name, { group: NAME });
            W.accelerators.events.register(name, null, () => buttons[btn].callback(buttons[btn]));
            W.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; }' +
          'button.waze-btn.E95-J { display: none }'
        );
    }

    // Bootstrap plugin
    function bootstrap(tries = 1) {
        log('attempt ' + tries);
        if (W &&
          W.map &&
          W.model &&
          W.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);

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址