Greasy Fork is available in English.

WME Street View Availability

Shows a layer displaying the available street view roads and locations

// ==UserScript==
// @name        WME Street View Availability
// @namespace   http://www.tomputtemans.com/
// @description Shows a layer displaying the available street view roads and locations
// @include     /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor.*$/
// @icon        
// @version     1.0.1
// @grant       none
// ==/UserScript==

/* global W, I18n, OL, google */

(function() {
  var tilelayerServers = [
    'https://mts0.google.com/mapslt',
    'https://mts1.google.com/mapslt',
    'https://mts2.google.com/mapslt',
    'https://mts3.google.com/mapslt'
  ];

  function init(e) {
    if (e && e.user === null) {
      return;
    }
    if (typeof I18n === 'undefined') {
      setTimeout(init, 300);
      return;
    }
    if (document.getElementById('layer-switcher') === null && document.getElementById('layer-switcher-group_display') === null) {
      setTimeout(init, 200);
      return;
    }
    var streetViewControl = document.querySelector('.street-view-control');
    var buttons = document.getElementById('overlay-buttons-region');
    if (typeof streetViewControl === 'undefined' || typeof buttons === 'undefined') {
      setTimeout(init, 400);
      console.log('Street view control unavailable, retrying in 400ms');
    }
    if (typeof W === 'undefined' ||
        typeof W.loginManager === 'undefined') {
      setTimeout(init, 100);
      return;
    }
    if (!W.loginManager.user) {
      W.loginManager.events.register("login", null, init);
      if (!W.loginManager.user) {
        return;
      }
    }

    var enteringStreetView = false, // Set to true when the marker is being dragged to the map
        ignoreStreetViewExit = false, // Set to true to indicate that the street view availability was set to visible manually and should not be reverted
        pinWasHidden = true; // Contains previous pin display state, used to detected changes

    // Change the opacity with the following bookmarklet:
    // javascript:localStorage.WME_StreetViewAvailability=JSON.stringify({opacity:prompt('Give a percentage between 0 and 100',100)/100});W.map.getLayersByName('Street View Availability')[0].setOpacity(JSON.parse(localStorage.WME_StreetViewAvailability).opacity);

    // Add the map layer, hidden by default
    I18n.translations[I18n.currentLocale()].layers.name.street_view_availability = 'Street View Availability';
    var streetViewLayer = new OL.Layer.XYZ('Street View', tilelayerServers[Math.floor(Math.random() * tilelayerServers.length)] + '?lyrs=svv&x=${x}&y=${y}&z=${z}&w=256&h=256&style=40', {
      isBaseLayer: false,
      uniqueName: 'street_view_availability',
      tileSize: new OL.Size(256, 256),
      transitionEffect: 'resize',
      displayInLayerSwitcher: true,
      opacity: localStorage.WME_StreetViewAvailability ? JSON.parse(localStorage.WME_StreetViewAvailability).opacity : 1,
      visibility: false
    });
    W.map.addLayer(streetViewLayer);

    // Add layer entry in the new layer drawer
    var displayGroupToggle = document.getElementById('layer-switcher-group_display');
    if (displayGroupToggle != null) {
      var displayGroup = displayGroupToggle.parentNode;
      while (displayGroup != null && displayGroup.className != 'group') {
        displayGroup = displayGroup.parentNode;
      }
      var togglesList = displayGroup.querySelector('.collapsible-GROUP_DISPLAY');
      var toggler = document.createElement('li');
      var checkbox = document.createElement('wz-checkbox');
      checkbox.id = 'layer-switcher-item_street_view';
      checkbox.type = 'checkbox';
      checkbox.className = 'hydrated';
      checkbox.textContent = 'Street View';
      checkbox.addEventListener('click', e => streetViewLayer.setVisibility(e.target.checked));
      toggler.appendChild(checkbox);
      togglesList.appendChild(toggler);
      displayGroupToggle.addEventListener('click', function() {
        checkbox.disabled = !displayGroupToggle.checked;
        streetViewLayer.setVisibility(displayGroupToggle.checked && checkbox.checked);
      });
    }

    // Create keyboard shortcut to toggle the imagery layer (Shift+T)
    I18n.translations[I18n.currentLocale()].keyboard_shortcuts.groups.layers.members.toggleStreetViewAvailability = 'Toggle street view availability';
    W.accelerators.addAction('toggleStreetViewAvailability', { group: 'layers' });
    W.accelerators.events.register('toggleStreetViewAvailability', this, function() {
      streetViewLayer.setVisibility(!streetViewLayer.getVisibility());
      checkbox.checked = streetViewLayer.getVisibility();
    });
    W.accelerators._registerShortcuts({ 'S+t': 'toggleStreetViewAvailability' });

    // Add an observer to activate the script whenever the street view marker gets dragged around
    var controlObserver = new MutationObserver(function(mutationRecords) {
      try {
        var activeButton = mutationRecords.find(record => record.target.classList.contains('overlay-button-active'));
        if ((activeButton == null) != pinWasHidden) {
          if (pinWasHidden == true && displayGroupToggle.checked) {
            pinWasHidden = activeButton == null;
            enteringStreetView = true;
            streetViewLayer.setVisibility(true);
            enteringStreetView = false;
          } else if (pinWasHidden == false && !ignoreStreetViewExit) {
            pinWasHidden = activeButton == null;
            streetViewLayer.setVisibility(false);
          }
        }
      } catch (error) {
        console.error('Error caught while observing pin mutation', error);
      }
    });
    controlObserver.observe(streetViewControl, { attributes: true, attributeFilter: ['class'] });
    var buttonsObserver = new MutationObserver(function(mutationRecords) {
      mutationRecords.forEach(record => {
        var streetViewControl = record.target.querySelector('.street-view-control');
        if (streetViewControl) {
          controlObserver.observe(streetViewControl, { attributes: true, attributeFilter: ['class'] });
        }
      });
    });
    buttonsObserver.observe(buttons, { childList: true });

    // Deal with changes to the layer visibility
    streetViewLayer.events.register('visibilitychanged', null, function() {
      if (!enteringStreetView && streetViewLayer.getVisibility()) {
        ignoreStreetViewExit = true;
      }
      if (!streetViewLayer.getVisibility()) {
        ignoreStreetViewExit = false;
      }
    });
  }
  init();
})();