// ==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);