// ==UserScript==
// @name WME E87 Inconsistent direction
// @name:uk WME 🇺🇦 E87 Inconsistent direction
// @version 0.1.0
// @description Solves the inconsistent direction problem
// @description:uk Дозволяє вирішувати проблему різнонаправленних сегментів
// @license MIT License
// @author Anton Shevchuk
// @namespace https://greasyfork.org/users/227648-anton-shevchuk
// @supportURL https://github.com/AntonShevchuk/wme-template/issues
// @match https://*.waze.com/editor*
// @match https://*.waze.com/*/editor*
// @exclude https://*.waze.com/user/editor*
// @icon 
// @grant none
// @require https://update.greasyfork.org/scripts/389765/1090053/CommonUtils.js
// @require https://update.greasyfork.org/scripts/450160/1218867/WME-Bootstrap.js
// @require https://update.greasyfork.org/scripts/452563/1218878/WME.js
// @require https://update.greasyfork.org/scripts/450221/1137043/WME-Base.js
// @require https://update.greasyfork.org/scripts/450320/1281847/WME-UI.js
// ==/UserScript==
/* jshint esversion: 8 */
/* global require */
/* global $, jQuery */
/* global W */
/* global I18n */
/* global OpenLayers */
/* global WME, WMEBase */
/* global WMEUI, WMEUIHelper, WMEUIHelperPanel, WMEUIHelperModal, WMEUIHelperTab, WMEUIShortcut, WMEUIHelperFieldset */
/* global Container, Settings, SimpleCache, Tools */
(function () {
'use strict'
// Script name, uses as unique index
const NAME = 'E87'
// Translations
const TRANSLATION = {
'en': {
title: 'Direction →',
description: 'Plugin WME E87 solves the inconsistent direction problem.<br/>Choose one or more segment to change direction.',
buttons: {
toggle: 'Change direction',
forward: 'A → B',
reverse: 'B → A',
},
},
'uk': {
title: 'Напрямки →',
description: 'Плагін WME E87 для вирішиння проблеми різно направленних вулиць.<br/>Оберіть один або декілька сегментів щоб застосувати зміни.',
buttons: {
toggle: 'Змінити напрямок',
forward: 'A → B',
reverse: 'B → A',
},
},
'ru': {
title: 'Направления →',
description: 'Плагин WME E87 для решения проблемы разнонаправленных улиц.<br/>Выберите один или несколько сегментов, чтобы внести изменения.',
buttons: {
toggle: 'Изменить направление',
forward: 'A → B',
reverse: 'B → A',
},
}
}
const STYLE =
'.lanes-tab div.e87 { border: 1px solid var(--hairline); border-radius: 6px; margin-bottom: 16px; padding: 8px 16px 18px; } ' +
'button.waze-btn.e87 { background: #f2f4f7; border: 1px solid #ccc; margin: 2px; } ' +
'button.waze-btn.e87:hover { background: #ffffff; transition: background-color 100ms linear; 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.e87:focus { background: #f2f4f7; } ' +
'button.e87-forward, button.e87-reverse { margin: 2px 8px; }' +
'div.e87-container { display: flex; flex: auto; justify-content: space-evenly; } ' +
'p.e87-info { border-top: 1px solid #ccc; color: #777; font-size: x-small; margin-top: 15px; padding-top: 10px; text-align: center; }'
WMEUI.addTranslation(NAME, TRANSLATION)
WMEUI.addStyle(STYLE)
const BUTTONS = {
toggle: {
title: I18n.t(NAME).buttons.toggle,
description: I18n.t(NAME).buttons.toggle,
shortcut: '',
},
}
// Default settings
const SETTINGS = {}
class E87 extends WMEBase {
constructor (name, settings = null) {
super(name, settings)
/** @type {WMEUIHelper} */
this.helper = new WMEUIHelper(this.name)
/** @type {WMEUIHelperTab} */
this.tab = this.helper.createTab(I18n.t(this.name).title, { image: GM_info.script.icon })
this.tab.addText('description', I18n.t(this.name).description)
this.tab.addText('info', '<a href="' + GM_info.scriptUpdateURL + '">' + GM_info.script.name + '</a> ' + GM_info.script.version)
this.tab.inject()
/** @type {WMEUIHelperPanel} */
this.panel = this.helper.createPanel(I18n.t(name).title)
}
/**
* Init button for selection of the segment
* @param buttons
*/
init (buttons) {
buttons.toggle.callback = (e) => {
e.preventDefault()
WME.getSelectedSegments().forEach(
segment => this.invert(segment.getID())
)
}
this.panel.addButtons(buttons)
}
/**
* Handler for `segment.wme` event
* @param {jQuery.Event} event
* @param {HTMLElement} element
* @param {W.model} model
* @return {void}
*/
onSegment (event, element, model) {
// Skip for walking trails and blocked roads
if (model.isWalkingRoadType()
|| model.isLockedByHigherRank()
|| !model.isGeometryEditable()) {
return
}
element
//.parentNode.parentNode
//.querySelector('.lanes-tab')
.prepend(this.panel.html())
}
/**
* Handler for `segments.wme` event
* @param {jQuery.Event} event
* @param {HTMLElement} element
* @param {Array} models
* @return {void}
*/
onSegments (event, element, models) {
// Skip for walking trails or locked roads
if (models.filter((model) => model.isWalkingRoadType() || model.isLockedByHigherRank() || !model.isGeometryEditable()).length > 0) {
element.querySelector('div.form-group.e87')?.remove()
return
}
let reversed = W.selectionManager.getReversedSegments()
if (reversed.numReversed === 0) {
// you can reverse all selected segments
element
//.parentNode.parentNode
//.querySelector('.lanes-tab')
.prepend(this.panel.html())
return
}
let result = this.detect(reversed)
if (result.forward.length && result.reverse.length) {
this.log('Inconsistent direction detected: forward = ' + result.forward.length + ' backward = ' + result.reverse.length)
let buttonToForward = document.createElement('button')
buttonToForward.type = 'button'
buttonToForward.title = I18n.t(NAME).buttons.toggle
buttonToForward.className = 'waze-btn waze-btn-small waze-btn-white e87 e87-forward'
buttonToForward.innerText = I18n.t(NAME).buttons.forward + ' (' + result.reverse.length + ')'
buttonToForward.onclick = (e) => {
e.preventDefault()
result.reverse.forEach(el => this.invert(el))
buttonToForward.innerText = I18n.t(NAME).buttons.forward + ' (0)'
buttonToForward.disabled = true
}
let buttonToReverse = document.createElement('button')
buttonToReverse.type = 'button'
buttonToReverse.title = I18n.t(NAME).buttons.toggle
buttonToReverse.className = 'waze-btn waze-btn-small waze-btn-white e87 e87-reverse'
buttonToReverse.innerText = I18n.t(NAME).buttons.reverse + ' (' + result.forward.length + ')'
buttonToReverse.onclick = (e) => {
e.preventDefault()
result.forward.forEach(el => this.invert(el))
buttonToReverse.innerText = I18n.t(NAME).buttons.reverse + ' (0)'
buttonToReverse.disabled = true
}
this.container?.remove();
this.container = document.createElement('div')
this.container.className = 'e87-container'
this.container.append(buttonToForward)
this.container.append(buttonToReverse)
$('wz-alert.sidebar-alert.inconsistent-direction-alert .sidebar-alert-content')
.after(this.container)
}
}
/**
* Detect directions
* @param {Object} segments information
* @return {Object}
*/
detect (segments) {
let forward = [], reverse = []
for (let el in segments) {
el = Number.parseInt(el)
if (Number.isNaN(el)) {
continue
}
if (segments[el]) {
reverse.push(el)
} else {
forward.push(el)
}
}
return {
forward: forward,
reverse: reverse
}
}
/**
* Invert direction of the segment
* @param {Number} id of the segment
*/
invert (id) {
let segment = W.model.segments.getObjectById(id)
if (segment.isLockedByHigherRank()) {
this.log('Locked by higher rank')
return
}
this.group('invert segment ' + id)
this.log('segment', segment)
// setup and reverse attributes
let attributes = {}
attributes.fwdDirection = segment.attributes.revDirection
attributes.revDirection = segment.attributes.fwdDirection
let fwdTurnsLocked = segment.attributes.fwdTurnsLocked
let revTurnsLocked = segment.attributes.revTurnsLocked
// attributes.fwdTurnsLocked = segment.attributes.revTurnsLocked // ???
// attributes.revTurnsLocked = segment.attributes.fwdTurnsLocked // ???
// segment.setAttribute("revTurnsLocked", segment.attributes.fwdTurnsLocked)}
// segment.setAttribute("fwdTurnsLocked", segment.attributes.revTurnsLocked)}
attributes.fwdMaxSpeed = segment.attributes.revMaxSpeed
attributes.revMaxSpeed = segment.attributes.fwdMaxSpeed
attributes.fwdMaxSpeedUnverified = segment.attributes.revMaxSpeedUnverified
attributes.revMaxSpeedUnverified = segment.attributes.fwdMaxSpeedUnverified
attributes.fwdLaneCount = segment.attributes.revLaneCount
attributes.revLaneCount = segment.attributes.fwdLaneCount
attributes.restrictions = []
for (let i = 0; i < segment.attributes.restrictions.length; i++) {
attributes.restrictions[i] = segment.attributes.restrictions[i].withReverseDirection()
}
this.log('attributes', attributes)
let fromNode = segment.getFromNode()
let toNode = segment.getToNode()
let onA = {}
let toConnections = {}
fromNode.getSegmentIds().forEach(segId => {
// incoming directions
if (segId !== id) {
onA[segId] = W.model.getTurnGraph().getTurnThroughNode(fromNode, W.model.segments.getObjectById(segId), segment)
onA[segId].toVertex.direction = onA[segId].toVertex.direction === 'fwd' ? 'rev' : 'fwd'
}
// outgoing directions
toConnections[segId] = W.model.getTurnGraph().getTurnThroughNode(fromNode, segment, W.model.segments.getObjectById(segId))
toConnections[segId].fromVertex.direction = toConnections[segId].fromVertex.direction === 'fwd' ? 'rev' : 'fwd'
// u-turn
if (segId === id) {
toConnections[segId].toVertex.direction = toConnections[segId].toVertex.direction === 'fwd' ? 'rev' : 'fwd'
}
})
let onB = {}
let fromConnections = {}
toNode.getSegmentIds().forEach(segId => {
if (segId !== id) {
onB[segId] = W.model.getTurnGraph().getTurnThroughNode(toNode, W.model.segments.getObjectById(segId), segment)
onB[segId].toVertex.direction = onB[segId].toVertex.direction === 'fwd' ? 'rev' : 'fwd'
}
fromConnections[segId] = W.model.getTurnGraph().getTurnThroughNode(toNode, segment, W.model.segments.getObjectById(segId))
fromConnections[segId].fromVertex.direction = fromConnections[segId].fromVertex.direction === 'fwd' ? 'rev' : 'fwd'
// u-turn
if (segId === id) {
fromConnections[segId].toVertex.direction = fromConnections[segId].toVertex.direction === 'fwd' ? 'rev' : 'fwd'
}
})
// invert the geometry of the segment
let geometry = segment.getOLGeometry().clone()
geometry.components.reverse()
if (!geometry.components[0].equals(toNode.getOLGeometry())) {
let delta = { x: 0, y: 0 }
delta.x = toNode.getOLGeometry().x - geometry.components[0].x
delta.y = toNode.getOLGeometry().y - geometry.components[0].y
geometry.components[0].move(delta.x, delta.y)
}
let points = geometry.components.length - 1
if (!geometry.components[points].equals(fromNode.getOLGeometry())) {
let delta = { x: 0, y: 0 }
delta.x = fromNode.getOLGeometry().x - geometry.components[points].x
delta.y = fromNode.getOLGeometry().y - geometry.components[points].y
geometry.components[points].move(delta.x, delta.y)
}
// disconnect the segment
let disconnect = new WazeActionMultiAction([new WazeActionDisconnectSegment(segment, fromNode), new WazeActionDisconnectSegment(segment, toNode)])
disconnect._description = I18n.t('save.changes_log.actions.DisconnectSegment.default')
W.model.actionManager.add(disconnect)
// update geometry of the segment
W.model.actionManager.add(new WazeActionUpdateSegmentGeometry(segment, segment.getGeometry(), W.userscripts.toGeoJSONGeometry(geometry)))
// update attributes
W.model.actionManager.add(new WazeActionUpdateObject(segment, attributes))
// connect the segment
let connect = new WazeActionMultiAction([new WazeActionConnectSegment(toNode, segment), new WazeActionConnectSegment(fromNode, segment)])
connect._description = I18n.t('save.changes_log.actions.ConnectSegment.default')
W.model.actionManager.add(connect)
// update Turn's attributes
segment.setAttribute('fwdTurnsLocked', revTurnsLocked)
segment.setAttribute('revTurnsLocked', fwdTurnsLocked)
// W.model.actionManager.add(new WazeActionUpdateObject(segment, segment.getAttributes()))
// allow all connections
// W.model.actionManager.add(new WazeActionModifyAllConnections(segment.getToNode(), true));
// W.model.actionManager.add(new WazeActionModifyAllConnections(segment.getFromNode(), true));
this.applyTurns(fromConnections)
this.applyTurns(toConnections)
this.applyTurns(onA)
this.applyTurns(onB)
this.groupEnd()
}
/**
* Apply turns for segments
* @param segments
*/
applyTurns (segments) {
let actions = []
for (let sid in segments) {
let segment = segments[sid]
let turn
switch (segment.turnData.state) {
case 0 :
case 1 :
turn = WazeModelGraphTurnData.create()
turn = turn.withState(segment.turnData.state)
.withRestrictions(segment.turnData.restrictions)
.withInstructionOpcode(segment.turnData.instructionOpcode)
.withLanes(segment.turnData.lanes)
actions.push(new WazeModelGraphActionsSetTurn(W.model.getTurnGraph(), segment.withTurnData(turn)))
break
}
}
let multiAction = new WazeActionMultiAction(actions)
multiAction._description = I18n.t('save.changes_log.actions.SetTurn.update')
W.model.actionManager.add(multiAction)
}
}
let WazeActionConnectSegment
let WazeActionDisconnectSegment
let WazeActionModifyAllConnections
let WazeActionMultiAction
let WazeActionUpdateObject
let WazeActionUpdateSegmentGeometry
let WazeModelGraphTurnData
let WazeModelGraphActionsSetTurn
$(document).on('bootstrap.wme', () => {
let Instance = new E87(NAME, SETTINGS)
Instance.init(BUTTONS)
WazeActionConnectSegment = require('Waze/Action/ConnectSegment')
WazeActionDisconnectSegment = require('Waze/Action/DisconnectSegment')
WazeActionModifyAllConnections = require('Waze/Action/ModifyAllConnections')
WazeActionMultiAction = require('Waze/Action/MultiAction')
WazeActionUpdateObject = require('Waze/Action/UpdateObject')
WazeActionUpdateSegmentGeometry = require('Waze/Action/UpdateSegmentGeometry')
WazeModelGraphTurnData = require('Waze/Model/Graph/TurnData')
WazeModelGraphActionsSetTurn = require('Waze/Model/Graph/Actions/SetTurn')
})
})()