WME Reselect

Utility to redo recent selections

// ==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://greasyfork.org/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);
}