WME Reselect

Utility to redo recent selections

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==UserScript==
// @name           WME Reselect
// @description    Utility to redo recent selections
// @namespace      [email protected]
// @grant          none
// @grant          GM_info
// @version        0.0.1
// @include 	     /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor.*$/
// @exclude        https://www.waze.com/user/*editor/*
// @exclude        https://www.waze.com/*/user/*editor/*
// @author         GyllieGyllie
// @license        MIT/BSD/X11
// @require        https://greatest.deepsurf.us/scripts/24851-wazewrap/code/WazeWrap.js
// ==/UserScript==
/* Changelog

*/
/* global W */
/* global I18n */
/* global $ */

// Credits for icon go to FlatIcon
const undoIcon = '';

const ScriptName = GM_info.script.name;
const ScriptVersion = GM_info.script.version;

let ChangeLog = "WME Reselect has been updated to " + ScriptVersion + "<br />";
ChangeLog = ChangeLog + "<br /><b>New: </b>";
ChangeLog = ChangeLog + "<br />" + "- Ability to redo your recent selections";

let wmeSDK;
const options = loadOptions();

const selectionHistory = [];
let rollbackCount = 0;
let rollbackPending = undefined;

// Now validate the options are ok
validateOptions(options);

function log(message) {
  if (typeof message === 'string') {
    console.log('Reselect: ' + message);
  } else {
    console.log('Reselect: ', message);
  }
}

// the sdk init function will be available after the WME is initialized
function WMEReselect_bootstrap() {
  if (!wmeSDK.DataModel.Countries.getTopCountry() || !WazeWrap.Ready) {
    setTimeout(WMEReselect_bootstrap, 250);
    return;
  }

  if (wmeSDK.State.isReady) {
    WMEReselect_init();
  } else {
    wmeSDK.Events.once({ eventName: "wme-ready" }).then(WMEReselect_init);
  }
}

function WMEReselect_init() {
  log("Start");

  constructSettings();
  displayChangelog();

  const button = document.createElement('div');
  button.style.cssText = 'cursor:pointer;float:left;height:20px;width:20px;margin:5px;background-image: url(\''+ undoIcon + '\');background-size:contain;';

  button.onclick = () =>  doRollback();
  $('.secondary-toolbar-actions:not(.user-toolbar)').prepend(button);

  wmeSDK.Events.on({
    eventName: 'wme-selection-changed',
    eventHandler: addSelection
  });

  log("Done");

}

// Check if unsafeWindow is availabe, if so use that
('unsafeWindow' in window ? window.unsafeWindow : window).SDK_INITIALIZED.then(() => {
  // initialize the sdk with your script id and script name
  wmeSDK = getWmeSdk({scriptId: "wme-reselect", scriptName: "Reselect"});
  WMEReselect_bootstrap();
});

function displayChangelog() {
  if (!WazeWrap.Interface) {
    setTimeout(displayChangelog, 1000);
    return;
  }

  // Alert the user version updates
  if (options.lastAnnouncedVersion === ScriptVersion) {
    log('Version: ' + ScriptVersion);
  } else {
    WazeWrap.Interface.ShowScriptUpdate(ScriptName, ScriptVersion, ChangeLog + "<br /><br />", "https://github.com/wazers/wme-reselect");

    const updateName = "#wmereselect" + ScriptVersion.replaceAll(".", "");
    $(updateName + " .WWSUFooter a").text("Github")

    options.lastAnnouncedVersion = ScriptVersion;
    saveOptions(options);
  }
}

function addSelection() {
  const selection = wmeSDK.Editing.getSelection();
  console.log(selection);

  if (!selection) {
    if (!rollbackPending) {
      // Only reset when no pending rollback
      // Sometimes the set selection does an unselect first?
      rollbackCount = 0;
    }
    return;
  }

  if (rollbackPending && selection.objectType === rollbackPending.objectType && selection.ids.length === rollbackPending.ids.length) {
    rollbackPending = null;
    return;
  }

  // We don't track this selection type
  if (options.activeSelectionTypes.indexOf(selection.objectType) === -1) {
    rollbackCount = 0;
    return;
  }

  // New selection so remove all rolled back selections
  if (rollbackCount > 1) {
    selectionHistory.splice(selectionHistory.length - rollbackCount);
    rollbackCount = 0;
  }

  selectionHistory.push(selection);
  log("count: " + selectionHistory.length);
  log("rollback: " + rollbackCount);

  // History is becoming to large so remove oldest element
  if (selectionHistory.length > options.maxHistory) {
    selection.splice(0, 1);
  }
}

function doRollback() {
  const currentSelection = wmeSDK.Editing.getSelection();

  rollbackCount += 1;

  if (rollbackCount > selectionHistory.length) {
    rollbackCount = selectionHistory.length;
  }

  rollbackPending = selectionHistory[selectionHistory.length - rollbackCount];

  if (rollbackCount === 0 && !!currentSelection && currentSelection.objectType === rollbackPending.objectType) {
    // Selection was still active so we need to ignore current selection for rollback
    rollbackCount = 1;
    rollbackPending = selectionHistory[selectionHistory.length - rollbackCount];
  }

  log("rollback: " + rollbackCount);
  log(rollbackPending);

  wmeSDK.Editing.setSelection({
    selection: rollbackPending
  });
}

////////////////////////////////////////////////////////////////////////////////////////////////////////////
////
//// Option Logic
////
////////////////////////////////////////////////////////////////////////////////////////////////////////////
function constructSettings() {

  // -- Set up the tab for the script
  wmeSDK.Sidebar.registerScriptTab().then(({ tabLabel, tabPane }) => {
    tabLabel.innerText = 'Reselect';
    tabLabel.title = 'Reselect';

    tabPane.innerHTML = '<div id="reselect-settings"></div>';

    const scriptContentPane = $('#reselect-settings');

    scriptContentPane.append(`<h2 style="margin-top: 0;">Reselect</h2>`);
    scriptContentPane.append(`<span>Current Version: <b>${ScriptVersion}</b></span>`);

    addTextNumberSettings(scriptContentPane, 'To avoid memory issues during long edit sessions do not make this too large', 'Max history', 'maxHistory');

    scriptContentPane.append(`<h5>Selection Types To Track:</h5>`);

    addBooleanListSettingsCallback(scriptContentPane, '', 'Segments', 'activeSelectionTypes', 'segment')
  });

}

function toggleTrackOptions(event, value) {
  const current = options['activeSelectionTypes'].indexOf(value);

  if (current >= 0) {
    options['activeSelectionTypes'].splice(current, 1);
  } else {
    options['activeSelectionTypes'].push(value);
  }
  saveOptions(options);
}

function getDefaultOptions() {
  return {
    lastAnnouncedVersion: '',
    maxHistory: 100,
    activeSelectionTypes: [
      'segment'
    ]
  }
}

function loadOptions() {
  let text = localStorage.getItem("Reselect-Options");
  let options;

  if (text) {
    options = JSON.parse(text);
  } else {
    options = getDefaultOptions();
  }

  return options;
}

function validateOptions(options) {
  const defaultOptions = getDefaultOptions();

  // Add missing options
  for (let key in defaultOptions) {
    if (!(key in options)) {
      options[key] = defaultOptions[key]
    }
  }
}

function saveOptions(options) {
  const optionsJson = JSON.stringify(options);
  localStorage.setItem("Reselect-Options", optionsJson);
}

function changeText(event) {
  options[event.target.id] = event.target.value;
  saveOptions(options);
}

function addTextNumberSettings(container, title, label, name, step = 1) {
  const currentValue = options[name];

  const textInput = $('<wz-text-input type="number" min="0" max="999" step="' + step + '" id="' + name + '" value="' + currentValue + '"></wz-text-input>');
  const optionHtml = $('<div style="margin-top: 10px;"><span Title="' + title + '">' + label + '</span></div>').append(textInput);

  container.append(optionHtml);

  textInput.on('change', changeText);
}

function addBooleanSettingsCallback(container, title, label, name, clickHandler) {
  const currentValue = options[name];

  const checkbox = $('<wz-checkbox id="' + name + '" Title="' + title + '" name="types" disabled="false" checked="' + currentValue + '">' + label + '</wz-checkbox>');
  const optionHtml = $('<div class="reselect-option"></div>').append(checkbox);

  container.append(optionHtml);

  checkbox.on('click', clickHandler);
}

function addBooleanListSettingsCallback(container, title, label, name, value, clickHandler) {
  const currentlyActive = options[name].indexOf(value) >= 0;

  const checkbox = $('<wz-checkbox id="' + name + '" Title="' + title + '" name="types" disabled="false" checked="' + currentlyActive + '">' + label + '</wz-checkbox>');
  const optionHtml = $('<div class="reselect-option"></div>').append(checkbox);

  container.append(optionHtml);

  checkbox.on('click', clickHandler);
}

function toggleBoolean(event) {
  options[event.target.id] = event.target.checked;
  saveOptions(options);
}