WME Image Overlays

Makes it possible to add images as overlay on the Waze Map Editor

// ==UserScript==
// @name        WME Image Overlays
// @author      Tom 'Glodenox' Puttemans
// @namespace   http://www.tomputtemans.com/
// @description Makes it possible to add images as overlay on the Waze Map Editor
// @include     /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor.*$/
// @icon        
// @version     1.5.2
// @grant       none
// ==/UserScript==

/* global W, $, OpenLayers, I18n */

var styleElement;

async function onWmeReady() {
  enrichOpenLayers();

  var io_strings = {
    en: {
      tab_title: 'Image Overlays',
      warning_title: 'Warning! Make sure you know which data sources you may use!',
      warning_explanation1: 'As this script allows you to add external sources to add information to the map, it is important to know that not all sources can be used freely by Waze.',
      warning_explanation2: 'Copying information from data sources like OpenStreetMap and Bing Maps is <strong>strictly forbidden</strong>. It may even get you banned from editing the map as it could require Waze to remove certain sections from the map.',
      safe_sources_header: 'These are usually safe sources to use:',
      safe_sources_list: '<li>Information sent to you directly through mail by an organiser</li><li>Information on a governmental website</li><li>Maps found in newspapers or news sites</li><li>Data released as "open data" with no restrictions on commercial usage</li>',
      community_advice: 'When in doubt about a resource, contact <a href="https://wazeopedia.waze.com/">your local community</a> and ask for their advice.',
      warning_button: "I'll keep this in mind",
      warning_info: 'Note: this is a one-time only warning',
      add_image: 'Add image overlay',
      empty_list: 'No images added yet',
      name_missing: 'No name specified',
      hide_overlay: 'Hide overlay',
      opacity: 'Opacity:',
      parent_map_layer: 'Parent map layer',
      parent_map_layer_help: 'This decides on top of which map layer the image overlay will be drawn',
      layer_hidden: 'Hidden',
      image_name: 'Overlay Image Name',
      stretch_image: 'Stretch:',
      control_sensitivity: 'Sensitivity:',
      persistent_warning: 'Data is currently not stored persistently, your browser may remove image overlays over time without warning.',
      persistent_warning_link: 'Read more...',
      export_image: 'Export image',
      import_image: 'Import image',
      import_image_description: 'You can now paste an image from your clipboard in the WME with Ctrl+V or select an image with the file input field below.',
      import_error: 'Could not import image, the image is probably too big to retrieve. If you used the clipboard, you may want to download the image and try the file input field above instead.',
      align_image: 'Align with map',
      align_image_description: "You can use the controls below to align the image overlay with the map. Use the 'Attach to map' button to finish.",
      attach_image: 'Attach to map',
      cancel: 'Cancel'
    }
  };
  setTranslations(io_strings);
  applyStyles();

  var tab = await addTab();
  var imagesList = document.createElement('div'),
      overlayControls = document.createElement('div');

  if (localStorage.ImageOverlays_warning == undefined && W.loginManager.getUserRank() < 3) {
    imagesList.classList.add('hidden');
    overlayControls.classList.add('hidden');
    var warningMessage = document.createElement('div');
    warningMessage.className = 'image-overlays-source-warning';
    var exclamationMark = document.createElement('i');
    exclamationMark.className = 'fa fa-fw fa-3x fa-exclamation-triangle pull-left';
    warningMessage.appendChild(exclamationMark);
    var warningTitle = document.createElement('h3');
    warningTitle.textContent = I18n.t('image_overlays.warning_title');
    warningMessage.appendChild(warningTitle);
    var warningExplanation1 = document.createElement('p');
    warningExplanation1.textContent = I18n.t('image_overlays.warning_explanation1');
    warningMessage.appendChild(warningExplanation1);
    var warningExplanation2 = document.createElement('p');
    warningExplanation2.innerHTML = I18n.t('image_overlays.warning_explanation2');
    warningMessage.appendChild(warningExplanation2);
    var safeSourcesHeader = document.createElement('p');
    safeSourcesHeader.style.marginBottom = '0';
    safeSourcesHeader.textContent = I18n.t('image_overlays.safe_sources_header');
    warningMessage.appendChild(safeSourcesHeader);
    var safeSourcesList = document.createElement('ul');
    safeSourcesList.innerHTML = I18n.t('image_overlays.safe_sources_list');
    warningMessage.appendChild(safeSourcesList);
    var communityAdvice = document.createElement('p');
    communityAdvice.innerHTML = I18n.t('image_overlays.community_advice');
    warningMessage.appendChild(communityAdvice);
    var warningButton = document.createElement('button');
    warningButton.className = 'btn btn-success center-block';
    warningButton.disabled = true;
    warningButton.addEventListener('click', function() {
      warningMessage.classList.add('hidden');
      imagesList.classList.remove('hidden');
      overlayControls.classList.remove('hidden');
      localStorage.ImageOverlays_warning = 'accepted';
    });
    var checkSign = document.createElement('i');
    checkSign.className = 'fa fa-fw fa-check';
    warningButton.appendChild(checkSign);
    warningButton.appendChild(document.createTextNode(I18n.t('image_overlays.warning_button')));
    var countdownSeparator = document.createTextNode(' - ');
    warningButton.appendChild(countdownSeparator);
    var warningCountdown = document.createElement('span');
    warningCountdown.textContent = '20';
    warningButton.appendChild(warningCountdown);
    warningMessage.appendChild(warningButton);
    var warningInfo = document.createElement('footer');
    warningInfo.textContent = I18n.t('image_overlays.warning_info');
    warningMessage.appendChild(warningInfo);
    tab.appendChild(warningMessage);
    var tabActivityListener = new MutationObserver(function() {
      if (tab.parentNode.classList.contains('active')) {
        var intervalId = setInterval(function() {
          if (warningCountdown.textContent > 1) {
            warningCountdown.textContent = warningCountdown.textContent - 1;
          } else {
            clearInterval(intervalId);
            warningButton.disabled = false;
            warningButton.removeChild(countdownSeparator);
            warningButton.removeChild(warningCountdown);
          }
        }, 1000);
        tabActivityListener.disconnect();
      }
    });
    tabActivityListener.observe(tab.parentNode, { attributes: true, attributeFilter: [ 'class' ] });
  }

  var layer = null,
      currentBlob = null,
      currentKey = null;

  var editPanel = document.createElement('div');
  editPanel.className = 'hidden';

  var importError = document.createElement('p');
  importError.className = 'hidden text-danger';
  importError.textContent = I18n.t('image_overlays.import_error');
  var pasteListener = function(e) {
    var items = e.clipboardData.items;
    for (var i = 0; i < items.length; ++i) {
      if (items[i].kind == 'file' && items[i].type.indexOf('image/') !== -1) {
        var blob = items[i].getAsFile();
        if (blob) {
          importError.classList.add('hidden');
          displayAlignPage({
            blob: blob
          });
        } else {
          importError.classList.remove('hidden');
        }
        break;
      }
    }
  };

  var persistentStorageWarning = document.createElement('div');
  persistentStorageWarning.className = 'alert alert-danger text-center hidden';
  persistentStorageWarning.style.marginBottom = '1em';
  persistentStorageWarning.innerHTML = '<i class="fa fa-exclamation-triangle"></i> ' + I18n.t('image_overlays.persistent_warning') + ' <a href="https://www.waze.com/forum/viewtopic.php?f=819&t=225760&p=1850215#p1850215" target="_blank">' + I18n.t('image_overlays.persistent_warning_link') + '</a>';
  tab.appendChild(persistentStorageWarning);

  var cancelButton,
      cancelButtonIcon = document.createElement('i'),
      panelTitle = document.createElement('h3'),
      panelTitleIcon = document.createElement('i'),
      panelTitleText = document.createElement('span'),
      description = document.createElement('p'),
      instructions = document.createElement('div'),
      addImageOverlayButton = document.createElement('button'),
      addSpan = document.createElement('span'),
      importButton = document.createElement('button'),
      hideOverlayButton = document.createElement('button'),
      layerControls = document.createElement('div'),
      opacityRange = document.createElement('input'),
      opacityLabel = document.createElement('label'),
      imageNameInput = document.createElement('wz-text-input'),
      exportButton = document.createElement('button');
  panelTitle.appendChild(panelTitleIcon);
  panelTitle.appendChild(document.createTextNode(' '));
  panelTitle.appendChild(panelTitleText);
  panelTitle.className = 'image-overlays-panel-title';
  editPanel.appendChild(panelTitle);
  description.className = 'text-center';
  editPanel.appendChild(description);
  instructions.className = 'text-center';
  editPanel.appendChild(instructions);
  tab.appendChild(editPanel);

  var emptyList = document.createElement('span');
  emptyList.style.fontStyle = 'italic';
  emptyList.appendChild(document.createTextNode(I18n.t('image_overlays.empty_list')));
  tab.appendChild(emptyList);
  imagesList.classList.add('result-list');
  imagesList.style.marginBottom = '1em';
  tab.appendChild(imagesList);
  getIndexedDB(function(db) {
    db.transaction(['overlays'], 'readonly')
      .objectStore('overlays')
      .openCursor()
      .addEventListener('success', function(e) {
      var cursor = e.target.result;
      if (cursor) {
        addImageOverlay(cursor.value.name, cursor.key);
        cursor.continue();
      }
    });
  });

  addImageOverlayButton.className = 'btn btn-primary';
  addSpan.className = 'fa fa-plus';
  addSpan.style.marginRight = '5px';
  addImageOverlayButton.appendChild(addSpan);
  addImageOverlayButton.appendChild(document.createTextNode(I18n.t('image_overlays.add_image')));
  addImageOverlayButton.addEventListener('click', displayImportPage);
  overlayControls.appendChild(addImageOverlayButton);
  importButton.className = 'btn btn-default';
  importButton.style.float = 'right';
  importButton.innerHTML = '<i class="fa fa-download"></i> ' + I18n.t('image_overlays.import_image');
  importButton.addEventListener('click', importLayer);
  overlayControls.appendChild(importButton);
  hideOverlayButton.className = 'btn btn-default hidden';
  hideOverlayButton.style.float = 'right';
  hideOverlayButton.textContent = I18n.t('image_overlays.hide_overlay');
  hideOverlayButton.addEventListener('click', removeLayer);
  overlayControls.appendChild(hideOverlayButton);
  tab.appendChild(overlayControls);

  layerControls.className = 'hidden clearfix controls';
  opacityRange.type = 'range';
  opacityRange.min = 0;
  opacityRange.max = 50;
  opacityRange.value = 50;
  opacityRange.id = 'imageOverlaysOpacity';
  opacityLabel.textContent = I18n.t('image_overlays.opacity');
  opacityLabel.htmlFor = opacityRange.id;
  layerControls.appendChild(opacityLabel);
  layerControls.appendChild(opacityRange);
  var rangeListener = function() {
    if (layer && layer.key) {
      getIndexedDB(function(db) {
        var objectStore = db.transaction(['overlays'], 'readwrite').objectStore('overlays');
        objectStore.get(layer.key).addEventListener('success', function(e) {
          var overlay = e.target.result;
          overlay.opacity = opacityRange.value / 50;
          objectStore.put(overlay, layer.key).addEventListener('success', function() {
            layer.setOpacity(opacityRange.value / 50);
          });
        });
      });
    } else if (layer) {
      layer.setOpacity(opacityRange.value / 50);
    }
  };
  opacityRange.addEventListener('input', rangeListener);
  opacityRange.addEventListener('change', rangeListener);
  var parentLayer = document.createElement('select');
  parentLayer.className = 'form-control';
  parentLayer.id = 'imageOverlaysParentLayer';
  W.map.events.on({
    addlayer: updateParentLayer,
    removelayer: updateParentLayer,
    changelayer: updateParentLayer
  });
  parentLayer.addEventListener('change', function() {
    if (layer && layer.key) {
      getIndexedDB(function(db) {
        var objectStore = db.transaction(['overlays'], 'readwrite').objectStore('overlays');
        objectStore.get(layer.key).addEventListener('success', function(e) {
          var overlay = e.target.result;
          overlay.layerTarget = parentLayer.value;
          objectStore.put(overlay, layer.key).addEventListener('success', moveImageOverlayLayer);
        });
      });
    } else if (layer) {
      moveImageOverlayLayer();
    }
  });
  var parentLayerLabel = document.createElement('label');
  parentLayerLabel.textContent = I18n.t('image_overlays.parent_map_layer') + ' ';
  parentLayerLabel.htmlFor = parentLayer.id;
  parentLayerLabel.style.marginTop = '10px';
  var parentLayerHelp = document.createElement('i');
  parentLayerHelp.className = 'waze-tooltip';
  parentLayerHelp.title = I18n.t('image_overlays.parent_map_layer_help');
  $(parentLayerHelp).tooltip();
  parentLayerLabel.appendChild(parentLayerHelp);
  layerControls.appendChild(parentLayerLabel);
  layerControls.appendChild(parentLayer);
  exportButton.className = 'btn btn-default';
  exportButton.style.margin = '10px 0';
  exportButton.innerHTML = '<i class="fa fa-upload"></i> ' + I18n.t('image_overlays.export_image');
  exportButton.addEventListener('click', exportLayer);
  layerControls.appendChild(exportButton);
  tab.appendChild(layerControls);
  imageNameInput.className = 'imageoverlays-margins';
  imageNameInput.placeholder = I18n.t('image_overlays.image_name');

  var editButtonsContainer = document.createElement('div');
  editButtonsContainer.className = 'hidden imageoverlays-margins';
  editButtonsContainer.style.height = '30px';
  cancelButton = document.createElement('button');
  cancelButton.className = 'btn btn-default cancel';
  cancelButtonIcon.className = 'fa fa-trash-o fa-fw';
  cancelButton.appendChild(cancelButtonIcon);
  cancelButton.appendChild(document.createTextNode(I18n.t('image_overlays.cancel')));
  cancelButton.addEventListener('click', function() {
    editPanel.classList.add('hidden');
    editButtonsContainer.classList.add('hidden');
    exportButton.classList.remove('hidden');
    overlayControls.classList.remove('hidden');
    imagesList.classList.remove('hidden');
    emptyList.classList.toggle('hidden', imagesList.childNodes.length > 0);
    currentKey = null;
    removeLayer();
  });
  editButtonsContainer.appendChild(cancelButton);
  var pinToMapButton = document.createElement('button');
  pinToMapButton.className = 'btn btn-primary';
  pinToMapButton.innerHTML = '<i class="fa fa-map-pin fa-fw"></i> ' + I18n.t('image_overlays.attach_image');
  pinToMapButton.addEventListener('click', pinToMap);
  editButtonsContainer.appendChild(pinToMapButton);
  tab.appendChild(editButtonsContainer);

  var versionBlock = document.createElement('p');
  var versionInfo = document.createElement('a');
  versionInfo.appendChild(document.createTextNode(GM_info.script.name + ' (v' + GM_info.script.version + ')'));
  versionInfo.href = 'https://www.waze.com/forum/viewtopic.php?f=819&t=225760';
  versionInfo.target = '_blank';
  versionBlock.appendChild(versionInfo);
  tab.appendChild(versionBlock);

  function addImageOverlay(name, key, selected) {
    emptyList.classList.add('hidden');
    var overlayHandle = document.createElement('wz-card');
    overlayHandle.className = 'list-item-card';
    overlayHandle.dataset.key = key;
    if (selected) {
      overlayHandle.style.fontWeight = '700';
    }
    var container = document.createElement('div');
    container.className = 'image-overlay-handle';
    var nameContainer = document.createElement('div');
    if (name && name.length > 0) {
      nameContainer.textContent = name;
    } else {
      nameContainer.style.fontStyle = 'italic';
      nameContainer.textContent = I18n.t('image_overlays.name_missing');
    }
    container.appendChild(nameContainer);
    var editOverlay = document.createElement('button');
    editOverlay.className = 'fa fa-pencil rename';
    editOverlay.addEventListener('click', function(e) {
      e.stopPropagation();
      getIndexedDB(function(db) {
        var objectStore = db.transaction(['overlays'], 'readwrite').objectStore('overlays');
        objectStore.get(key).addEventListener('success', function(e) {
          var overlay = e.target.result;
          overlay.key = key;
          editPanel.classList.remove('hidden');
          overlayControls.classList.add('hidden');
          editButtonsContainer.classList.remove('hidden');
          imagesList.classList.add('hidden');
          emptyList.classList.add('hidden');
          displayAlignPage(overlay, key);
        });
      });
    });
    container.appendChild(editOverlay);
    var remove = document.createElement('button');
    remove.className = 'fa fa-trash-o remove';
    remove.addEventListener('click', function(e) {
      e.stopPropagation();
      getIndexedDB(function(db) {
        db.transaction(['overlays'], 'readwrite').objectStore('overlays').delete(key).addEventListener('success', function(e) {
          overlayHandle.parentNode.removeChild(overlayHandle);
          emptyList.classList.toggle('hidden', imagesList.childNodes.length > 0);
          removeLayer();
        });
      });
    });
    container.appendChild(remove);
    container.addEventListener('click', function() {
      if (layer && layer.key && layer.key == key) {
        removeLayer();
      } else {
        getIndexedDB(function(db) {
          db.transaction(['overlays'], 'readonly').objectStore('overlays').get(key).addEventListener('success', function(e) {
            var overlay = e.target.result;
            overlay.key = key;
            displayImageOverlay(overlay);
          });
        });
      }
    });
    overlayHandle.appendChild(container);
    imagesList.appendChild(overlayHandle);
  }

  function updateParentLayer(currentLayer) {
    if (!currentLayer || typeof currentLayer == 'object') {
      currentLayer = parentLayer.value;
    }
    while (parentLayer.firstChild) {
      parentLayer.removeChild(parentLayer.firstChild);
    }
    let aboveRoadsOption = document.createElement('option');
    aboveRoadsOption.value = 'under-roads-layer';
    aboveRoadsOption.textContent = 'Under Roads';
    aboveRoadsOption.selected = aboveRoadsOption.value == currentLayer;
    parentLayer.appendChild(aboveRoadsOption);
    W.map.layers.forEach(function(layer) {
      if (layer.name != 'Image Overlay') {
        var layerOption = document.createElement('option');
        layerOption.value = layer.name;
        layerOption.textContent = layer.name + (layer.visibility ? '' : ' (' + I18n.t('image_overlays.layer_hidden') + ')');
        layerOption.selected = layer.name == currentLayer;
        parentLayer.appendChild(layerOption);
      }
    });
  }

  function displayImportPage() {
    editPanel.classList.remove('hidden');
    overlayControls.classList.add('hidden');
    editButtonsContainer.classList.remove('hidden');
    pinToMapButton.classList.add('hidden');
    imagesList.classList.add('hidden');
    emptyList.classList.add('hidden');
    document.addEventListener('paste', pasteListener);
    panelTitleIcon.className = 'fa fa-download';
    panelTitleText.textContent = I18n.t('image_overlays.import_image');
    removeLayer();

    description.textContent = I18n.t('image_overlays.import_image_description');
    var addImageInput = document.createElement('input');
    addImageInput.type = 'file';
    addImageInput.accepts = 'image/*';
    addImageInput.className = 'center-block';
    addImageInput.addEventListener('change', function() {
      displayAlignPage({
        blob: addImageInput.files[0]
      });
    });
    instructions.textContent = '';
    instructions.appendChild(addImageInput);
    instructions.appendChild(importError);

    if (navigator.storage.persist) {
      navigator.storage.persist().then((persistent) => {
        persistentStorageWarning.classList.toggle('hidden', persistent);
      });
    }
    editPanel.classList.remove('hidden');
  }

  function displayAlignPage(overlay, key) {
    currentBlob = overlay.blob;
    currentKey = key;
    document.removeEventListener('paste', pasteListener);
    panelTitleIcon.className = 'fa fa-arrows-alt';
    panelTitleText.textContent = I18n.t('image_overlays.align_image');
    parentLayer.selectedIndex = 0;
    exportButton.classList.add('hidden');
    pinToMapButton.classList.remove('hidden');

    displayImageOverlay(overlay, !currentKey);

    description.textContent = I18n.t('image_overlays.align_image_description');
    var scale = document.createElement('input');
    let numZoomLevels = W.map.getLayersBy("isBaseLayer", true)[0].numZoomLevels;
    instructions.textContent = '';
    instructions.appendChild(createControlButton('rotate-left', function() {
      layer.rotate(-45);
    }, '45°'));
    instructions.appendChild(createControlButton('rotate-left', function() {
      layer.rotate(-0.5 * scale.value/100);
    }));
    instructions.appendChild(createControlButton('arrow-up', function() {
      layer.shift(0, 10 * W.map.getResolution() * scale.value/100);
    }));
    instructions.appendChild(createControlButton('rotate-right', function() {
      layer.rotate(0.5 * scale.value/100);
    }));
    instructions.appendChild(createControlButton('rotate-right', function() {
      layer.rotate(45);
    }, '45°'));
    instructions.appendChild(document.createElement('br'));
    instructions.appendChild(createControlButton('arrow-left', function() {
      layer.shift(-10 * W.map.getResolution() * scale.value/100, 0);
    }));
    instructions.appendChild(createControlButton('crosshairs', function() {
      var layerCenter = layer.extent.getCenterLonLat();
      layer.shift(W.map.getCenter().lon - layerCenter.lon, W.map.getCenter().lat - layerCenter.lat);
    }));
    instructions.appendChild(createControlButton('arrow-right', function() {
      layer.shift(10 * W.map.getResolution() * scale.value/100, 0);
    }));
    instructions.appendChild(document.createElement('br'));
    instructions.appendChild(createControlButton('compress', function() {
      layer.scale(1 - 0.01 * (numZoomLevels-W.map.getZoom()) * scale.value/100);
    }));
    instructions.appendChild(createControlButton('arrow-down', function() {
      layer.shift(0, -10 * W.map.getResolution() * scale.value/100);
    }));
    instructions.appendChild(createControlButton('expand', function() {
      layer.scale(1 + 0.01 * (numZoomLevels-W.map.getZoom()) * scale.value/100);
    }));
    instructions.appendChild(document.createElement('br'));
    var stretchLabel = document.createElement('span');
    stretchLabel.textContent = I18n.t('image_overlays.stretch_image') + ' ';
    instructions.appendChild(stretchLabel);
    var horizontalStretchLabelIcon = document.createElement('i');
    horizontalStretchLabelIcon.className = 'fa fa-arrows-h';
    horizontalStretchLabelIcon.style.marginRight = '10px';
    instructions.appendChild(horizontalStretchLabelIcon);
    instructions.appendChild(createControlButton('plus', function() {
      layer.stretch(true, 1 + (0.01 * scale.value/100));
    }));
    instructions.appendChild(createControlButton('minus', function() {
      layer.stretch(true, 1 - (0.01 * scale.value/100));
    }));
    var verticalStretchLabelIcon = document.createElement('i');
    verticalStretchLabelIcon.style.margin = '15px 10px 15px 20px';
    verticalStretchLabelIcon.className = 'fa fa-arrows-v';
    verticalStretchLabelIcon.style.marginLeft = '10px';
    instructions.appendChild(verticalStretchLabelIcon);
    instructions.appendChild(createControlButton('plus', function() {
      layer.stretch(false, 1 + (0.01 * scale.value/100));
    }));
    instructions.appendChild(createControlButton('minus', function() {
      layer.stretch(false, 1 - (0.01 * scale.value/100));
    }));
    var sensitivityContainer = document.createElement('div');
    sensitivityContainer.className = 'imageoverlays-sensitivity';
    scale.id = 'imageoverlays-control-sensitivity';
    var controlSensitivity = document.createElement('label');
    controlSensitivity.textContent = I18n.t('image_overlays.control_sensitivity');
    controlSensitivity.htmlFor = scale.id;
    sensitivityContainer.appendChild(controlSensitivity);
    var sensitivity = document.createElement('span');
    sensitivity.textContent = '100%';
    scale.type = 'range';
    scale.min = 0;
    scale.value = 100;
    scale.max = 200;
    scale.addEventListener('input', function() {
      sensitivity.textContent = scale.value + '%';
    });
    sensitivityContainer.appendChild(scale);
    sensitivityContainer.appendChild(sensitivity);
    instructions.appendChild(sensitivityContainer);
    imageNameInput.value = overlay.name || currentBlob.name;
    instructions.appendChild(imageNameInput);
  }

  function pinToMap() {
    var obj = {
      'blob': currentBlob,
      'name': imageNameInput.value,
      'extent': layer.extent.toArray(),
      'rotation': layer.rotation,
      'opacity': opacityRange.value / 50,
      'layerTarget': parentLayer.value
    };
    storeOverlay(obj, function(e) {
      editPanel.classList.add('hidden');
      exportButton.classList.remove('hidden');
      overlayControls.classList.remove('hidden');
      imagesList.classList.remove('hidden');
      for (var i = imagesList.childNodes.length - 1; i >= 0; i--) {
        if (imagesList.childNodes[i].dataset.key == currentKey) {
          imagesList.removeChild(imagesList.childNodes[i]);
        }
      }
      layer.key = e.target.result;
      addImageOverlay(obj.name, e.target.result, true);
      currentKey = null;
    }, currentKey);
    editButtonsContainer.classList.add('hidden');
  }

  function storeOverlay(overlay, callback) {
    getIndexedDB(function(db) {
      if (currentKey) {
        db.transaction(['overlays'], 'readwrite').objectStore('overlays').put(overlay, currentKey).addEventListener('success', callback);
      } else {
        db.transaction(['overlays'], 'readwrite').objectStore('overlays').add(overlay).addEventListener('success', callback);
      }
    });
  }

  function displayImageOverlay(overlay, rescale) {
    var url = window.URL.createObjectURL(overlay.blob);
    overlay.extent = overlay.extent ? new OpenLayers.Bounds(overlay.extent) : W.map.getExtent();
    overlay.rotation = overlay.rotation || 0;
    var img = document.createElement('img');
    img.addEventListener('load', function() {
      removeLayer();
      if (rescale) { // Rescale the extent for the image so it has the correct aspect ratio
        var mapExtentAspectRatio = overlay.extent.getWidth() / overlay.extent.getHeight();
        var imageAspectRatio = img.naturalWidth / img.naturalHeight;
        if (mapExtentAspectRatio > imageAspectRatio) {
          var widthDiff = overlay.extent.getWidth() - (overlay.extent.getHeight() * imageAspectRatio);
          overlay.extent = new OpenLayers.Bounds([overlay.extent.left + widthDiff/2 , overlay.extent.bottom, overlay.extent.right - widthDiff/2, overlay.extent.top]);
        } else {
          var heightDiff = overlay.extent.getHeight() - (overlay.extent.getWidth() / imageAspectRatio);
          overlay.extent = new OpenLayers.Bounds([overlay.extent.left, overlay.extent.bottom + heightDiff/2, overlay.extent.right, overlay.extent.top - heightDiff/2]);
        }
        overlay.extent = overlay.extent.scale(0.8);
      }
      layer = new OpenLayers.Layer.OverlayImage('Image Overlay', url, overlay.extent, new OpenLayers.Size(img.naturalWidth, img.naturalHeight), overlay.key, { 'rotation': overlay.rotation, 'opacity': overlay.opacity || 1 });
      W.map.addLayer(layer);
      for (var i = 0; i < imagesList.childNodes.length; i++) {
        imagesList.childNodes[i].style.fontWeight = (imagesList.childNodes[i].dataset.key == overlay.key ? '700' : '');
      }
      layerControls.classList.remove('hidden');
      hideOverlayButton.classList.remove('hidden');
      importButton.classList.add('hidden');
      opacityRange.value = (overlay.opacity ? overlay.opacity * 50 : 50);
      updateParentLayer(overlay.layerTarget);
      if (parentLayer.value == 'under-roads-layer') {
        let targetIndex = W.map.getLayerIndex(W.map.getLayerByUniqueName('roads'));
        W.map.getOLMap().setLayerIndex(layer, targetIndex);
      } else {
        let targetIndex = W.map.getLayerIndex(W.map.getLayersByName(overlay.layerTarget || "")[0]);
        if (!targetIndex || targetIndex <= 0) {
          targetIndex = W.map.getLayerIndex(W.map.getLayerByUniqueName('roads'));
          W.map.getOLMap().setLayerIndex(layer, targetIndex);
        } else {
          W.map.getOLMap().setLayerIndex(layer, targetIndex + 1);
        }
      }
      if (!W.map.getExtent().intersectsBounds(overlay.extent)) {
        W.map.getOLMap().zoomToExtent(overlay.extent);
      }
    });
    img.addEventListener('error', e => log(e));
    img.src = url;
  }

  function moveImageOverlayLayer() {
    let targetIndex = null;
    if (parentLayer.value == 'under-roads-layer') {
      targetIndex = W.map.getLayerIndex(W.map.getLayerByUniqueName('roads')) - 1;
    } else {
      targetIndex = W.map.getLayerIndex(W.map.getLayersByName(parentLayer.value)[0]) + 1;
    }
    W.map.getOLMap().setLayerIndex(layer, targetIndex);
  }

  function removeLayer() {
    if (layer) {
      W.map.removeLayer(layer);
      layer = null;
      layerControls.classList.add('hidden');
      hideOverlayButton.classList.add('hidden');
      importButton.classList.remove('hidden');
      for (var i = 0; i < imagesList.childNodes.length; i++) {
        imagesList.childNodes[i].style.fontWeight = '';
      }
    }
  }

  function exportLayer() {
    if (layer && layer.key) {
      getIndexedDB(function(db) {
        db.transaction(['overlays'], 'readonly').objectStore('overlays').get(layer.key).addEventListener('success', function(e) {
          var downloadPrompt = function(data) {
            var download = document.createElement('a');
            download.download = data.name.replace(/[/\\?%*:|"<>\.$#,= ]/g, '-') + '.json'; // Transform all commonly reserved file name characters to dashes
            download.style.display = 'none';
            download.rel = 'noopener';
            var bytes = new TextEncoder().encode(JSON.stringify(data));
            download.href = URL.createObjectURL(new Blob([bytes], { type: "application/json;charset=utf-8" }));
            document.body.appendChild(download);
            download.click();
            setTimeout(function() { URL.revokeObjectURL(download.href) }, 30000);
            document.body.removeChild(download);
          };
          var result = e.target.result;
          if (result.blob) {
            var fileReader = new FileReader();
            fileReader.addEventListener('load', function() {
              result.blob = fileReader.result;
              downloadPrompt(result);
            });
            fileReader.readAsDataURL(result.blob);
          } else {
            downloadPrompt(result);
          }
        });
      });
    }
  }

  // Request file to import, then process it
  function importLayer() {
    var fileInput = document.createElement('input');
    fileInput.type = 'file';
    fileInput.accept = '.json';
    fileInput.style.display = 'none';
    fileInput.addEventListener('change', function() {
      var fileReader = new FileReader();
      fileReader.addEventListener('load', function() {
        document.body.removeChild(fileInput);
        var result = JSON.parse(fileReader.result);
        if (result.blob) {
          var mimeTypeSearch = result.blob.match(/:([^;]+);/);
          var mimeType = (mimeTypeSearch.length == 2 ? mimeTypeSearch[2] : '');
          var bytes = atob(result.blob.split(',')[1]);
          var arrayBuffer = new ArrayBuffer(bytes.length);
          var intArray = new Uint8Array(arrayBuffer);
          for (var i = 0; i < bytes.length; i++) {
            intArray[i] = bytes.charCodeAt(i);
          }
          result.blob = new File([ arrayBuffer ], result.name, { type: mimeType });
        }
        storeOverlay(result, function(e) {
          removeLayer();
          addImageOverlay(result.name, e.target.result, true);
          result.key = e.target.result;
          displayImageOverlay(result);
        });
      });
      fileReader.readAsText(fileInput.files[0]);
    });
    document.body.appendChild(fileInput);
    fileInput.click();
  }
}

function enrichOpenLayers() {
  if (OpenLayers.Layer.OverlayImage) {
    return;
  }
  // Based on the OpenLayers.Layer.Image class
  OpenLayers.Layer.OverlayImage = OpenLayers.Class(OpenLayers.Layer, {
    isBaseLayer: false,
    url: null,
    extent: null,
    size: null,
    tile: null,
    rotation: null,
    key: null,
    initialize: function(name, url, extent, size, key, options) {
      options = options || {};
      this.url = url;
      this.extent = extent;
      this.maxExtent = extent;
      this.size = size;
      this.key = key;
      this.rotation = options.rotation || 0;
      OpenLayers.Layer.prototype.initialize.apply(this, [name, options]);
    },
    destroy: function() {
      if (this.tile) {
        this.removeTileMonitoringHooks(this.tile);
        this.tile.destroy();
        this.tile = null;
      }
      OpenLayers.Layer.prototype.destroy.apply(this, arguments);
    },
    setMap: function(map) {
      OpenLayers.Layer.prototype.setMap.apply(this, arguments);
    },
    moveTo:function(bounds, zoomChanged, dragging) {
      OpenLayers.Layer.prototype.moveTo.apply(this, arguments);
      var firstRendering = (this.tile == null);
      if (zoomChanged || firstRendering) {
        this.setTileSize();
        var ulPx = this.map.getLayerPxFromLonLat({
          lon: this.extent.left,
          lat: this.extent.top
        });

        if (firstRendering) {
          this.tile = new OpenLayers.Tile.Image(this, ulPx, this.extent, null, this.tileSize);
          this.addTileMonitoringHooks(this.tile);
        } else {
          this.tile.size = this.tileSize.clone();
          this.tile.position = ulPx.clone();
        }
        this.tile.draw();
        this.setRotation(this.rotation);
      }
    },
    shift: function(x, y) {
      this.extent = this.extent.add(x, y);
      var ulPx = this.map.getLayerPxFromLonLat({
        lon: this.extent.left,
        lat: this.extent.top
      });
      this.tile.position = ulPx.clone();
      this.tile.positionTile();
    },
    scale: function(factor) {
      this.extent = this.extent.scale(factor);
      this.setTileSize();
      var ulPx = this.map.getLayerPxFromLonLat({
        lon: this.extent.left,
        lat: this.extent.top
      });
      this.tile.position = ulPx.clone();
      this.tile.size = this.tileSize.clone();
      this.tile.positionTile();
    },
    stretch: function(horizontal, factor) {
      var extent = this.extent.toArray();
      var distance;
      if (horizontal) {
        distance = Math.abs(extent[2] - extent[0]) * (1-factor) / 2;
        extent[0] += distance;
        extent[2] -= distance;
      } else {
        distance = Math.abs(extent[1] - extent[3]) * (1-factor) / 2;
        extent[1] += distance;
        extent[3] -= distance;
      }
      this.extent = OpenLayers.Bounds.fromArray(extent);
      this.setTileSize();
      var ulPx = this.map.getLayerPxFromLonLat({
        lon: this.extent.left,
        lat: this.extent.top
      });
      this.tile.position = ulPx.clone();
      this.tile.size = this.tileSize.clone();
      this.tile.positionTile();
    },
    rotate: function(rotation) {
      this.setRotation(this.rotation + rotation);
    },
    setTileSize: function() {
      var tileWidth = this.extent.getWidth() / this.map.getResolution();
      var tileHeight = this.extent.getHeight() / this.map.getResolution();
      this.tileSize = new OpenLayers.Size(tileWidth, tileHeight);
    },
    addTileMonitoringHooks: function(tile) {
      tile.onLoadStart = function() {
        this.events.triggerEvent("loadstart");
      };
      tile.events.register("loadstart", this, tile.onLoadStart);
      tile.onLoadEnd = function() {
        this.events.triggerEvent("loadend");
      };
      tile.events.register("loadend", this, tile.onLoadEnd);
      tile.events.register("unload", this, tile.onLoadEnd);
    },
    removeTileMonitoringHooks: function(tile) {
      tile.unload();
      tile.events.un({
        "loadstart": tile.onLoadStart,
        "loadend": tile.onLoadEnd,
        "unload": tile.onLoadEnd,
        scope: this
      });
    },
    setUrl: function(newUrl) {
      this.url = newUrl;
      this.tile.draw();
    },
    getURL: function(bounds) {
      return this.url;
    },
    setRotation: function(rotation) {
      this.rotation = rotation;
      this.tile.getTile().style.transform = 'rotate(' + rotation + 'deg)';
    },
    CLASS_NAME: "OpenLayers.Layer.OverlayImage"
  });
}

function setTranslations(translations) {
  I18n.translations[I18n.currentLocale()].image_overlays = translations.en;
  for (var i = 0; i < Object.keys(translations).length; i++) {
    var locale = Object.keys(translations)[i];
    if (I18n.currentLocale() == locale) {
      I18n.translations[locale].image_overlays = translations[locale];
      return;
    }
  }
}

// Create the tab
async function addTab() {
  const {tabLabel, tabPane} = W.userscripts.registerSidebarTab("imageoverlays");

  tabLabel.innerHTML = '<span class="fa fa-picture-o"></span>';
  tabLabel.title = I18n.t('image_overlays.tab_title');
  tabPane.id = 'sidepanel-imageoverlays';

  await W.userscripts.waitForElementConnected(tabPane);

  return Promise.resolve(tabPane);
}

function createControlButton(icon, callback, text) {
  var controlButton = document.createElement('button');
  var controlButtonIcon = document.createElement('i');
  controlButtonIcon.className = 'fa fa-fw fa-' + icon;
  controlButton.appendChild(controlButtonIcon);
  if (text) {
    controlButton.appendChild(document.createTextNode(' ' + text));
  }
  controlButton.addEventListener('click', callback);
  return controlButton;
}

function getIndexedDB(callback) {
  var req = indexedDB.open('ImageOverlays', 1);
  req.addEventListener('upgradeneeded', function(e) {
    if (e.oldVersion) {
      log('upgradeneeded event triggered again? Old version: ' + e.oldVersion);
      log(e);
    }
    e.target.result.createObjectStore('overlays', { autoIncrement: true });
  });
  req.addEventListener('error', log);
  req.addEventListener('success', e => callback(e.target.result));
}

function log(message) {
  if (typeof message === 'string') {
    console.log('%c' + GM_info.script.name + ' (v' + GM_info.script.version + '): %c' + message, 'color:black', 'color:#d97e00');
  } else {
    console.log('%c' + GM_info.script.name + ' (v' + GM_info.script.version + ')', 'color:black', message);
  }
}

function applyStyles() {
  if (!styleElement) {
    styleElement = document.createElement('style');
    styleElement.textContent = `
.image-overlays-source-warning {
  background-color: #fff;
  z-index: 1;
  border-radius: 8px;
  border: 2px solid #f00;
  padding: 8px;
}

.image-overlays-source-warning > i:first-child {
  color: #f00;
}

.image-overlays-source-warning h3 {
  margin-bottom: 15px;
  font-size: 13px;
  line-height: 1.42857;
  color: #f00;
  font-weight: 700;
}

.image-overlays-source-warning ul {
  padding-left: 5%;
}

.image-overlays-source-warning footer {
  font-style: italic;
  text-align: center;
}

h3.image-overlays-panel-title {
  margin-bottom: 1.5em;
  text-align: center;
}

#sidepanel-imageoverlays .result-list {
  display: flex;
  flex-direction: column;
  gap: 5px;
}

#sidepanel-imageoverlays .hidden {
  display: none;
}

#sidepanel-imageoverlays .image-overlay-handle {
  display: flex;
  font-size: 14px;
  align-items: center;
  gap: 3px;
}

#sidepanel-imageoverlays .image-overlay-handle div:first-child {
  flex-grow: 1;
  word-wrap: anywhere;
}

#sidepanel-imageoverlays .cancel {
  position: absolute;
  right: 15px;
}

#sidepanel-imageoverlays .controls {
  margin-top: 8px;
}

#sidepanel-imageoverlays input[type=range] {
  width: 180px;
  display: inline;
  margin: 5px;
  vertical-align: bottom;
}

#sidepanel-imageoverlays .imageoverlays-sensitivity {
  display: flex;
  align-items: center;
}

#sidepanel-imageoverlays .imageoverlays-sensitivity label {
  margin-bottom: 0;
}

#sidepanel-imageoverlays #imageOverlaysParentLayer {
  background-color: var(--background_variant, #f2f4f7);
}

#sidepanel-imageoverlays p:last-child {
  font-size: 0.9em;
  margin-top: 10px;
}

#sidepanel-imageoverlays .result .remove {
  font-size: 14px;
  float: right;
  font-weight: normal;
  margin-top: -6px;
}

#sidepanel-imageoverlays .result .rename {
  font-size: 14px;
  float: right;
  font-weight: normal;
  margin-top: -6px;
  margin-left: 4px;
}

.imageoverlays-margins {
  margin: 15px 0 15px 0;
  position: relative;
}
`;
  }
  if (!styleElement.parentNode) {
    document.head.appendChild(styleElement);
  }
}

function onWmeInitialized() {
  if (W.userscripts?.state?.isReady) {
    log('W is ready and in "wme-ready" state. Proceeding with initialization.');
    onWmeReady();
  } else {
    log('W is ready, but not in "wme-ready" state. Adding event listener.');
    document.addEventListener('wme-ready', onWmeReady, { once: true });
  }
}

function bootstrap() {
  if (!W) {
    log('W is not available. Adding event listener.');
    document.addEventListener('wme-initialized', onWmeInitialized, { once: true });
  } else {
    onWmeInitialized();
  }
}

bootstrap();