// ==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://greasyfork.org/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://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @connect www.waze.com
// @connect greasyfork.org
// @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://greasyfork.org/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://greasyfork.org/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();
})();