WME E95

Setup road properties in one click

目前為 2019-05-17 提交的版本,檢視 最新版本

// ==UserScript==
// @name         WME E95
// @version      0.4.9
// @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    http://tampermonkey.net/
// ==/UserScript==
/* jshint esversion: 6 */
/* global require */

(function ($, WazeApi) {
  'use strict';

  let WazeActionUpdateObject = require("Waze/Action/UpdateObject");
  let WazeActionUpdateFeatureAddress = require("Waze/Action/UpdateFeatureAddress");

  // 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',
      keyCode: 49,
      detectCity: true,
      attributes: {
        fwdMaxSpeed: 5,
        revMaxSpeed: 5,
        roadType: types.parking,
        flags: 0,
        lockRank: 0,
      }
    },
    B: {
      title: 'Pr20',
      keyCode: 50,
      detectCity: true,
      attributes: {
        fwdMaxSpeed: 20,
        revMaxSpeed: 20,
        roadType: types.private,
        flags: 0,
        lockRank: 0,
      }
    },
    C: {
      title: 'Pr50',
      keyCode: 51,
      detectCity: true,
      attributes: {
        fwdMaxSpeed: 50,
        revMaxSpeed: 50,
        roadType: types.private,
        flags: 0,
        lockRank: 0,
      }
    },
    D: {
      title: 'St50',
      keyCode: 52,
      detectCity: true,
      attributes: {
        fwdMaxSpeed: 50,
        revMaxSpeed: 50,
        roadType: types.street,
        flags: 0,
        lockRank: 0,
      }
    },
    E: {
      title: 'PS50',
      keyCode: 53,
      detectCity: true,
      attributes: {
        fwdMaxSpeed: 50,
        revMaxSpeed: 50,
        roadType: types.primary,
        flags: 0,
        lockRank: 1,
      }
    },
    F: {
      title: 'OR',
      keyCode: 54,
      clearCity: true,
      attributes: {
        fwdMaxSpeed: 90,
        revMaxSpeed: 90,
        roadType: types.offroad,
        lockRank: 0,
      }
    },
    G: {
      title: 'Pr90',
      keyCode: 55,
      clearCity: true,
      attributes: {
        fwdMaxSpeed: 90,
        revMaxSpeed: 90,
        roadType: types.private,
        lockRank: 0,
      }
    },
    H: {
      title: 'St90',
      keyCode: 56,
      clearCity: true,
      attributes: {
        fwdMaxSpeed: 90,
        revMaxSpeed: 90,
        roadType: types.street,
        lockRank: 0,
      }
    },
    I: {
      title: 'PS90',
      keyCode: 57,
      clearCity: true,
      attributes: {
        fwdMaxSpeed: 90,
        revMaxSpeed: 90,
        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,
    }
  };

  // Get Button settings
  function getButtonConfig(index) {
    let btn = {};
    if (region[WazeApi.model.countries.top.abbr]
      && region[WazeApi.model.countries.top.abbr][index]) {
      $.extend(true, btn, buttons[index], region[WazeApi.model.countries.top.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.countries.top.id,
      stateID: addr.state ? addr.state.id : WazeApi.model.states.top.id,
      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) {
      console.log('E-95: city detection');
      let cityName = null;
      for (let i = 0, total = segments.length; i < total; i++) {
        console.log('E95: attempt ' + (i + 1));
        cityName = detectCity(segments[i]);
        if (cityName) {
          options.cityName = cityName;
          break;
        }
      }
      console.log('E-95: 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
    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 all buttons
    for (let btn in buttons) {
      let button = document.createElement('button');
      let buttonConfig = getButtonConfig(btn);
      button.className = 'waze-btn waze-btn-small road-e95 road-e95-' + btn;
      button.style.backgroundColor = colors[buttonConfig.attributes.roadType];
      button.innerHTML = buttonConfig.title;
      button.dataset.e95 = btn;
      controls.appendChild(button);
    }

    let label = document.createElement('label');
    label.className = 'control-label';
    label.innerHTML = 'Quick properties';

    let group = document.createElement('div');
    group.className = 'form-group e95';
    group.appendChild(label);
    group.appendChild(controls);

    document.getElementById('segment-edit-general').prepend(group);
  }

  function init() {
    // 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('div.hide-walking-trail').style.display !== 'none' &&
            !node.querySelector('div.form-group.e95')) {
            createUI();
          }
        }
      });
    });

    speedLimitsObserver.observe(document.getElementById('edit-panel'), {childList: true, subtree: true});
    console.log('E95: observer was run');

    // Handler for all buttons
    $('#edit-panel').on('click', 'button.road-e95', processHandler);

    // Handler for button shortcuts
    $(document).on('keyup', function (e) {
      if (e.altKey && !e.ctrlKey && !e.shiftKey) {
        for (let btn in buttons) {
          if (buttons[btn].keyCode === e.which) {
            process(btn);
            break;
          }
        }
      }
    });
    console.log('E95: handler was initialized');

    // Apply styles
    let style = document.createElement('style');
    style.type = 'text/css';
    style.innerHTML =
      'button.waze-btn.road-e95 { margin: 0 4px 4px 0; padding: 2px; width: 42px; } ' +
      'button.waze-btn.road-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.road-e95-E { margin-right: 42px; }' +
      'button.waze-btn.road-e95-F { margin-right: 50px; }'
    ;
    document.getElementsByTagName('head')[0].appendChild(style);
  }

  // Bootstrap plugin
  function bootstrap(tries = 1) {
    console.log('E95: attempt ' + tries);
    if (WazeApi &&
      WazeApi.map &&
      WazeApi.model &&
      WazeApi.loginManager.user) {
      console.log('E95: was initialized');
      init();
    } else if (tries < 100) {
      tries++;
      setTimeout(() => bootstrap(tries), 800);
    } else {
      console.error('E95: initialization failed');
    }
  }

  console.log('E95: initialization');
  bootstrap();
})(window.jQuery, window.W);

QingJ © 2025

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