您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Solves the inconsistent direction problem
当前为
// ==UserScript== // @name WME E87 Inconsistent direction // @version 0.0.2 // @description Solves the inconsistent direction problem // @license MIT License // @author Anton Shevchuk // @namespace https://gf.qytechs.cn/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://gf.qytechs.cn/scripts/389765-common-utils/code/CommonUtils.js?version=1090053 // @require https://gf.qytechs.cn/scripts/450160-wme-bootstrap/code/WME-Bootstrap.js?version=1128320 // @require https://gf.qytechs.cn/scripts/452563-wme/code/WME.js?version=1101598 // @require https://gf.qytechs.cn/scripts/450221-wme-base/code/WME-Base.js?version=1129908 // @require https://gf.qytechs.cn/scripts/450320-wme-ui/code/WME-UI.js?version=1128560 // ==/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', buttons: { toggle: 'Change direction', forward: 'A → B', reverse: 'B → A', }, }, 'uk': { title: 'Напрямки →', description: 'Плагін WME E87 для вирішиння проблеми різно направленних вулиць', buttons: { toggle: 'Змінити напрямок', forward: 'A → B', reverse: 'B → A', }, }, 'ru': { title: 'Направления →', description: 'Плагин WME E87 для решения проблемы разнонаправленных улиц', buttons: { toggle: 'Изменить направление', forward: 'A → B', reverse: 'B → A', }, } } const STYLE = '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) { super(name) /** @type {WMEUIHelper} */ this.helper = new WMEUIHelper(this.name) this.panel = this.helper.createPanel(I18n.t(name).title) /** @type {WMEUIHelperTab} */ this.tab = this.helper.createTab( I18n.t(this.name).title, I18n.t(this.name).description, { 'icon': '<i class="w-icon panel-header-component-icon w-icon-switch"></i>' } ) this.tab.addText( 'info', '<a href="' + GM_info.scriptUpdateURL + '">' + GM_info.script.name + '</a> ' + GM_info.script.version ) // Inject custom HTML to container in the WME interface this.tab.inject() } /** * Init button for selection of the segment * @param buttons */ init (buttons) { buttons.toggle.callback = (e) => { e.preventDefault() this.invert(WME.getSelectedSegment().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) { this.log('Selected one segment') element.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) { this.log('Check selected segments') let reversed = W.selectionManager.getReversedSegments() if (reversed.numReversed === 0) { 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.innerHTML = I18n.t(NAME).buttons.forward + ' (' + result.reverse.length + ')' buttonToForward.onclick = (e) => { e.preventDefault() result.reverse.forEach(el => this.invert(el)) } 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.innerHTML = I18n.t(NAME).buttons.reverse + ' (' + result.forward.length + ')' buttonToReverse.onclick = (e) => { e.preventDefault() result.forward.forEach(el => this.invert(el)) } let container = document.createElement('div') container.className = 'e87-container' container.append(buttonToForward) container.append(buttonToReverse) $('wz-alert.sidebar-alert.inconsistent-direction-alert .sidebar-alert-content') .after(container) } } /** * Detect directions * @param {Object} segments information * @return {Object} */ detect (segments) { let forward = [], reverse = [] for (let el in segments) { if (!Number.isNaN(Number.parseInt(el))) { 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 } console.groupCollapsed( '%c' + this.name + ':%c invert segment', 'color: #0DAD8D; font-weight: bold', 'color: dimgray; font-weight: normal' ) console.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() } console.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.geometry.clone() geometry.components.reverse() if (!geometry.components[0].equals(toNode.attributes.geometry)) { let delta = { x: 0, y: 0 } delta.x = toNode.attributes.geometry.x - geometry.components[0].x delta.y = toNode.attributes.geometry.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.attributes.geometry)) { let delta = { x: 0, y: 0 } delta.x = fromNode.attributes.geometry.x - geometry.components[points].x delta.y = fromNode.attributes.geometry.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.geometry, 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) console.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') }) })()
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址