您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
It counts the edits and suggests a time for the next save.
// ==UserScript== // @name WME Edition Counter and Helper // @name:es WME Contador de Ediciones y Ayudante // @description It counts the edits and suggests a time for the next save. // @description:es Cuentas las ediciones y sugiere un tiempo para el próximo guardado. // @author Rodrigo_Reina // @namespace https://gf.qytechs.cn/en/users/1192362-rodrigo-reina // @version 2023.10.11.03 // @license GNU GPLv3 // @icon  // @contributionURL https://ko-fi.com/wme_rodrigo_reina // @include /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor\/?.*$/ // @exclude https://www.waze.com/user/* // @exclude https://www.waze.com/*/user/* // @require https://ajax.googleapis.com/ajax/libs/jquery/2.1.4/jquery.min.js // @require https://gf.qytechs.cn/scripts/24851-wazewrap/code/WazeWrap.js // @connect www.waze.com // @connect gf.qytechs.cn // @grant GM_xmlhttpRequest // @grant GM_addElement // ==/UserScript== /* global W */ /* global toastr */ /* global WazeWrap */ /** * =============================================== * This script is based on the following scripts: * - "Waze Edit Count Monitor" (by MapOMatic) * - "Waze WME Edition Helper" (by EdwardNavarro) * =============================================== */ // TODO: // - Supporting languages (function main() { 'use strict'; const SCRIPT_NAME = 'WME Edition Counter and Helper'; const SCRIPT_VERSION = '2023.10.11.03'; const DOWNLOAD_URL = 'https://gf.qytechs.cn/scripts/40313-waze-edit-count-monitor/code/Waze%20Edit%20Count%20Monitor.user.js'; function wmeECHInjected() { const TOOLTIP_TEXT = '<b>Ediciones Diarias</b><br><small>(Clic para ver el perfil)<small>'; const TOASTR_URL = 'https://cdnjs.cloudflare.com/ajax/libs/toastr.js/2.1.4/toastr.min.js'; const VERSION = '2023.10.11.03'; const DEBUG_LEVEL = 0; let _toastrSettings = { timeBeforeSaving: 70, remindAtEditCount: 30, warnAtEditCount: 30, wasReminded: false, wasWarned: false }; let _userName = ''; let _lastTodayEditCount = 0; let _lastYesterdayEditCount = 0; let _lastDayBeforeEditCount = 0; let _savesWithoutIncrease = 0; let _totalSeconds = 0; let _timerInterval; let _buttonContainer, _buttonContentWrap, _buttonItemContainer, _buttonItemLink, _buttonItemContent, _progressBarWrap, _progressBarFill, _savedTimer; if (!localStorage.WMEEditionHelperScript) { let options = [null,_toastrSettings .timeBeforeSaving,_toastrSettings .remindAtEditCount,_toastrSettings .warnAtEditCount,false,false]; localStorage.WMEEditionHelperScript = JSON.stringify(options); } function log(message, level, prefix = 'LOG', bgColor = 'darkslategrey', textColor = 'white') { if (message && level <= DEBUG_LEVEL) { console.log('%c%s%s', `background:${bgColor};color:${textColor};padding:5px 10px;`, `[${prefix}] WME Edition Counter And Helper >>`, message); } } function checkCounters() { window.postMessage(JSON.stringify(['wme_echGetCounts', _userName]), '*'); _toastrSettings.wasReminded = false; _toastrSettings.wasWarned = false; toastr.remove(); } function getChangedObjectCount() { let _count = 0; try{ _count = parseInt($('#save-button > div.counter').text()); } catch(err){ // Do Nothing } return _count; } function updateEditCount(todayEditCount = 0, noIncrement) { let _textColor; let _bgColor; let _tooltipTextColor; if ($('#wme_ech').length === 0) { _buttonContainer = $('<div>', { id: 'wme_ech', style: 'border-radius: 15px;'}); _buttonContentWrap = $('<div>', { class: 'toolbar-button' }); _buttonItemLink = $('<a>', { href: 'https://www.waze.com/user/editor/' + _userName, target: '_blank', style:'text-decoration: none;', 'data-original-title': TOOLTIP_TEXT }); _buttonItemContainer = $('<div>', { class: 'item-container' }); _buttonItemContent = $('<div>', { style: 'margin: 7px; line-height: 1;' }); _progressBarWrap = $('<div>', { style: 'width: 100%; height: 5px; background-color: #d7dadc;border: 1px solid #fff; box-sizing: border-box;' }); _progressBarFill = $('<div>', { class: 'progress', style: 'width: 0%; height: 5px; animation-fill-mode: both; animation-name: progressBar; animation-duration:' + _toastrSettings.timeBeforeSaving + 's; animation-timing-function: linear;' }); _savedTimer = $('<div>', { id: 'saved-timer', style: 'font-size:8px; line-height:1; text-align:right; color:darkgray;' }); _buttonContainer.append(_buttonContentWrap); _buttonContentWrap.append(_buttonItemLink); _buttonItemLink.append(_buttonItemContainer); _buttonItemContainer.append(_buttonItemContent); _buttonItemContent.append(_progressBarWrap); _buttonItemContent.append(_savedTimer); _progressBarWrap.append(_progressBarFill); $('#toolbar > div > div.secondary-toolbar > div:nth-child(1)').after(_buttonContainer); _buttonItemLink.tooltip({ placement: 'auto top', delay: { show: 100, hide: 100 }, html: true, template: '<div class="tooltip" role="tooltip" style="opacity:0.95"><div class="tooltip-arrow"></div><div class="my-tooltip-header" style="display:block;"><b></b></div><div class="my-tooltip-body tooltip-inner" style="display:block;"></div></div>' }); } if (_lastTodayEditCount !== todayEditCount) { _savesWithoutIncrease = 0; } else { if (!noIncrement) _savesWithoutIncrease += 1; } switch (_savesWithoutIncrease) { case 0: case 1: _textColor = '#354148'; _bgColor = ''; _tooltipTextColor = 'white'; break; case 2: _textColor = '#354148'; _bgColor = 'yellow'; _tooltipTextColor = 'black'; break; default: _textColor = 'white'; _bgColor = 'red'; _tooltipTextColor = 'white'; } _buttonContainer.css('background-color', _bgColor); _buttonItemContent.css('color', _textColor).html(`Ediciones: ${todayEditCount}`); _buttonItemContent.append(_progressBarWrap); _buttonItemContent.append(_savedTimer); let _daysCounterText = `<hr style="border:0 none; border-bottom:1px #999 solid; margin:5px 0;"/><div class="days-group"><div class="day-1"><h3>Hoy</h3><span>${todayEditCount}</span></div><div class="day-2"><h3>Ayer</h3><span>${_lastYesterdayEditCount}</span></div><div class="day-3"><h3>Antier</h3><span>${_lastDayBeforeEditCount}</span></div></div>`; let _warningText = (_savesWithoutIncrease > 0) ? `<div style="font-size:13px;border-radius:5px;padding:5px;margin-top:5px;color:${_tooltipTextColor};background-color:${_bgColor};"><b>${_savesWithoutIncrease}</b> salvadas/guardadas consecutivas sin incremento en el contador.<br><span style="font-weight:bold;font-size:16px;">¿Estás estrangulado?<span></div>` : ''; _buttonItemLink.attr('data-original-title', TOOLTIP_TEXT + _daysCounterText + _warningText); _lastTodayEditCount = todayEditCount; _totalSeconds = 0; clearTimeout(_timerInterval); runTimer(); } function runTimer(){ _timerInterval = setInterval(setTime, 1000); } function setTime(){ ++_totalSeconds; const hours = parseInt(_totalSeconds / 3600) % 24; const minutes = parseInt(_totalSeconds / 60) % 60; const seconds = _totalSeconds % 60; const timeString = `${(hours > 0) ? `${pad(hours)}:` : ''}${pad(minutes)}:${pad(seconds)}`; $('#saved-timer').html(timeString); } function pad(val) { const valString = val + ''; return (valString.length < 2) ? '0' + valString : valString; } function receiveMessage(event) { let _msg; try { _msg = JSON.parse(event.data); } catch (err) { // Do nothing } if (_msg && _msg[0] === 'wme_echUpdateCounts') { const todayEditCount = _msg[1][0]; const yesterdayEditCount = _msg[1][1]; const dayBeforeEditCount = _msg[1][2]; _lastYesterdayEditCount = yesterdayEditCount; _lastDayBeforeEditCount = dayBeforeEditCount; updateEditCount(todayEditCount); } } function errorHandler(callback) { try { callback(); } catch (e) { console.error('%c%s%s', 'background:darkred;color:white;padding:5px 10px;', '[ERROR] WME Edition Counter And Helper >>', e); } } function checkChangedObjectCount() { let objectEditCount = getChangedObjectCount(); if (objectEditCount >= _toastrSettings.warnAtEditCount && !_toastrSettings.wasWarned) { toastr.remove(); toastr.error('<span style="font-size:16px;">Has editado al menos <b>' + _toastrSettings.warnAtEditCount + '</b> objetos.</span><br><br> Deberías considerar guardar pronto. Si obtienes un error al guardar, necesitarás deshacer algunos cambios/acciones e intentar nuevamente.', 'WME Edition Counter And Helper:', {timeOut: 25000}); _toastrSettings.wasWarned = true; //log('WARMED', 0, 'ALERT', 'tomato') } else if (objectEditCount >= _toastrSettings.remindAtEditCount && !_toastrSettings.wasReminded) { toastr.remove(); toastr.warning('<span style="font-size:16px;">Has editado al menos <b>' + _toastrSettings.remindAtEditCount + '</b> objetos.</span><br><br> Deberías considerar guardar pronto.', 'WME Edition Counter And Helper:', {timeOut: 15000}); _toastrSettings.wasReminded = true; //log('REMINDED', 0, 'ALERT', 'orange') } else if (objectEditCount < _toastrSettings.remindAtEditCount) { _toastrSettings.wasWarned = false; _toastrSettings.wasReminded = false; toastr.remove(); //log('REMOVED', 0, 'ALERT', 'sienna') } } /* helper functions */ function getElementsByClassName(classname, node) { if(!node) node = document.getElementsByTagName("body")[0]; let a = []; let re = new RegExp('\\b' + classname + '\\b'); let els = node.getElementsByTagName("*"); for (let i=0, j=els.length; i<j; i++) { if (re.test(els[i].className)) a.push(els[i]); } return a; } function getId(node) { return document.getElementById(node); } function updateAddonSettings(event) { _toastrSettings.timeBeforeSaving = getId('_ehSavingWaitTime').value; _toastrSettings.remindAtEditCount = getId('_ehRememberAfter').value; _toastrSettings.warnAtEditCount = getId('_ehAlertAfter').value; $('.progress').css('animation-duration', `${getId('_ehSavingWaitTime').value}s`); } async function init() { _userName = W.loginManager.user.getUsername(); window.addEventListener('message', receiveMessage); // restore saved settings if (localStorage.WMEEditionHelperScript) { let options = JSON.parse(localStorage.WMEEditionHelperScript); _toastrSettings.timeBeforeSaving = options[1]; _toastrSettings.remindAtEditCount = options[2]; _toastrSettings.warnAtEditCount = options[3]; } // check if sidebar is hidden let sidebar = getId('sidebar'); if (sidebar.style.display == 'none') { log("Not logged in yet - will initialise at login", 0, 'WARN', 'orange'); W.loginManager.events.register("login", null, init); return; } // check that user-info section is defined let userTabs = getId('user-info'); if (userTabs === null) { log("Editor not initialised yet - trying again in a bit...", 0, 'WARN', 'orange'); setTimeout(init, 789); return; } $('head').append( $('<link/>', { rel: 'stylesheet', type: 'text/css', href: 'https://cdnjs.cloudflare.com/ajax/libs/toastr.js/2.1.4/toastr.min.css' }), $('<style type="text/css">' + '#toast-container {position: absolute;} ' + '#toast-container > div {opacity: 0.95;} ' + '.toast-top-center {top: 30px;} ' + '#wme_ech { display:flex; } ' + '.toolbar .toolbar-icon-eh { color: #484848; font-size: 24px; margin: 8px 0; position: relative; text-align: center; width: 24px; } ' + '.progress { background-color: red; animation-fill-mode:both; } ' + '@keyframes progressBar { 0% { width: 0; } 99% { background-color: red; } 100% { width: 100%; background-color: green; } } ' + '.days-group { width:100%; display:flex; justify-content:space-between; align-item:center; } ' + '.days-group div { width:30%; padding:5px; background-color:darkgray; color:white; display:flex; flex-direction:column; align-item:center; border-radius:5px; } ' + '.days-group div h3 { font-size:12px; font-weight:bold; line-height:1; margin:5px 0; } ' + '.days-group div span { font-size:14px; font-weight:bold; } ' + '.days-group .day-1 { background-color:#22577a; } ' + '.days-group .day-2 { background-color:#38a3a5; } ' + '.days-group .day-3 { background-color:#57cc99; } ' + '</style>') ); await $.getScript(TOASTR_URL); toastr.options = { target: '#map', timeOut: 9999999999, positionClass: 'toast-top-right', closeOnHover: false, closeDuration: 0, showDuration: 0, closeButton: true }; W.model.actionManager.events.register('afterclearactions', null, () => errorHandler(checkCounters)); W.model.actionManager.events.register('afteraction', null, () => errorHandler(checkChangedObjectCount)); W.model.actionManager.events.register('afterundoaction', null, () => errorHandler(checkChangedObjectCount)); checkCounters(); toastr.success('Inicializado!', 'WME Edition Counter And Helper', {timeOut: 1500}); log('Initialized!', 0, 'SUCCESS', 'green'); //toastr.warning('<span style="font-size:16px;">Has editado al menos <b>' + _toastrSettings.remindAtEditCount + '</b> objetos.</span><br><br> Deberías considerar guardar pronto.', 'WME Edition Counter And Helper:', {timeOut: 1000}); // add new box to left of the map let navTabs = getElementsByClassName('nav-tabs', userTabs)[0]; let tabContent = getElementsByClassName('tab-content', userTabs)[0]; let addon = document.createElement('section'); addon.id = "edition-helper-addon"; // advanced options let section = document.createElement('p'); section.style.paddingTop = "0px"; section.className = 'checkbox'; section.id = 'advancedOptions'; section.innerHTML = '<h4><span class="fa fa-pencil" title="WME Edition Counter And Helper"></span> WME Edition Counter And Helper</h4><div style="margin:5px 0 10px 0;"><b>Configuración</b></div>' + '<label for="_ehSavingWaitTime">Tiempo de espera para guardar</label><br>' + '<input type="number" min="1" max="3600" size="4" id="_ehSavingWaitTime" style="margin: 0 0 20px 0" /> segundos' + '<br>' + '<label for="_ehRememberAfter">Recomendar guardar despues de</label><br>' + '<input type="number" min="1" max="5000" size="4" id="_ehRememberAfter" style="margin: 0 0 20px 0" /> cambios' + '<br>' + '<label for="_ehAlertAfter">Alertar guardar despues de</label><br>' + '<input type="number" min="1" max="5000" size="4" id="_ehAlertAfter" style="margin: 0 0 20px 0" /> cambios' ; addon.appendChild(section); // Addon legal and credits addon.innerHTML += '<hr style="border:0 none; border-bottom:1px #ccc solid;">' + '<small><b><a href="https://gf.qytechs.cn/en/scripts/434355-wme-edition-helper" target="_blank"><u>' + 'WME Edition Counter And Helper</u></a></b> v' + VERSION + '</small>'; // Add tab button and panel content let newtab = document.createElement('li'); newtab.innerHTML = '<a href="#sidepanel-edition-helper" data-toggle="tab"><span class="fa fa-pencil" title="WME Edition Counter And Helper"></span> WME ECH</a>'; navTabs.appendChild(newtab); addon.id = "sidepanel-edition-helper"; addon.className = "tab-pane"; tabContent.appendChild(addon); getId('_ehSavingWaitTime').onchange = updateAddonSettings; getId('_ehRememberAfter').onchange = updateAddonSettings; getId('_ehAlertAfter').onchange = updateAddonSettings; // restore saved settings if (localStorage.WMEEditionHelperScript) { let options = JSON.parse(localStorage.WMEEditionHelperScript); getId('_ehSavingWaitTime').value = options[1]; getId('_ehRememberAfter').value = options[2]; getId('_ehAlertAfter').value = options[3]; } // overload the WME exit function const saveEditionHelperOptions = function() { if (localStorage) { let options = []; // preserve previous options which may get lost after logout if (localStorage.WMEEditionHelperScript) { options = JSON.parse(localStorage.WMEEditionHelperScript); } options[1] = getId('_ehSavingWaitTime').value; options[2] = getId('_ehRememberAfter').value; options[3] = getId('_ehAlertAfter').value; localStorage.WMEEditionHelperScript = JSON.stringify(options); } } window.addEventListener("beforeunload", saveEditionHelperOptions, false); } function bootstrap() { if (W && W.loginManager && W.loginManager.events && W.loginManager.events.register && W.map && W.loginManager.user) { log('Initializing...'); init(); } else { log('Bootstrap failed. Trying again...'); setTimeout(bootstrap, 1000); } } bootstrap(); } function getEditCountFromProfile(profile) { const { editingActivity } = profile; return editingActivity[editingActivity.length - 1]; } function getEditCountByDayFromProfile(profile, day) { const { editingActivity } = profile; return editingActivity[editingActivity.length-day]; } function receivePageMessage(event) { let _msg; try { _msg = JSON.parse(event.data); } catch (err) { // Do Nothing } if (_msg && _msg[0] === 'wme_echGetCounts') { const userName = _msg[1]; GM_xmlhttpRequest({ method: 'GET', url: `https://www.waze.com/Descartes/app/UserProfile/Profile?username=${userName}`, onload: res => { const profile = JSON.parse(res.responseText); window.postMessage(JSON.stringify(['wme_echUpdateCounts', [ getEditCountFromProfile(profile), getEditCountByDayFromProfile(profile, 2), getEditCountByDayFromProfile(profile, 3) ]]), '*'); } }); } } function waitForWazeWrap() { return new Promise(resolve => { function loopCheck(tries = 0) { if (WazeWrap.Ready) { resolve(); } else if (tries < 1000) { setTimeout(loopCheck, 200, ++tries); } } loopCheck(); }); } function injectScript() { GM_addElement('script', { textContent: `${wmeECHInjected.toString()} \nwmeECHInjected();` }); window.addEventListener('message', receivePageMessage); } async function loadScriptUpdateMonitor() { let updateMonitor; await waitForWazeWrap(); try { updateMonitor = new WazeWrap.Alerts.ScriptUpdateMonitor(SCRIPT_NAME, SCRIPT_VERSION, DOWNLOAD_URL, GM_xmlhttpRequest); updateMonitor.start(); } catch (ex) { console.error(`${SCRIPT_NAME}:`, ex); } } function startFunction() { injectScript(); loadScriptUpdateMonitor(); } startFunction(); })();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址