WME OpenMaps

Add additional maps that are released as open data to the Waze Map Editor

目前為 2016-04-04 提交的版本,檢視 最新版本

// ==UserScript==
// @name        WME OpenMaps
// @author      Tom 'Glodenox' Puttemans
// @namespace   http://www.tomputtemans.com/
// @description Add additional maps that are released as open data to the Waze Map Editor
// @include     https://www.waze.com/*/editor/*
// @include     https://www.waze.com/editor/*
// @include     https://editor-beta.waze.com/*
// @exclude     https://www.waze.com/user/*editor/*
// @connect     wallonie.be
// @connect     informatievlaanderen.be
// @connect     agiv.be
// @connect     irisnet.be
// @connect     nationaalgeoregister.nl
// @connect     gbo-provincies.nl
// @icon        
// @supportURL  https://github.com/Glodenox/wme-om/issues
// @version     2.3.5
// @grant       GM_xmlhttpRequest
// ==/UserScript==
(function() {
  function init(e) {
    if (e && e.user == null) {
      return;
    }
    // Variables needed in their 'unsafe' context
    var OL = unsafeWindow.OL,
        Waze = unsafeWindow.Waze,
        I18n = unsafeWindow.I18n,
        $ = unsafeWindow.$;
    if (typeof Waze === 'undefined' || typeof Waze.map === 'undefined') {
      setTimeout(init, 800);
      log('Waze object unavailable, map still loading');
      return;
    }
    if (document.getElementById('user-info') == null) {
      setTimeout(init, 500);
      log('user-info element not yet available, map still loading');
      return;
    }
    if (typeof Waze.loginManager === 'undefined') {
      setTimeout(init, 300);
      return;
    }
    if (!Waze.loginManager.hasUser()) {
      Waze.loginManager.events.register("login", null, exportFunction(init, unsafeWindow));
      Waze.loginManager.events.register("loginStatus", null, exportFunction(init, unsafeWindow));
      return;
    }

    var versions = ['2.3.0', '2.3.1', '2.3.2', '2.3.3', '2.3.4', '2.3.5'];
    // set up language string
    var om_strings = {
      en: {
        tab_title: 'Open Maps',
        maps_title: 'Active Maps',
        no_local_maps: 'No maps found for this area',
        expand: 'Click to expand',
        collapse: 'Click to collapse',
        hideshow_layer: 'Hide/Show map',
        query_window_title: 'Map Location Query',
        query_window_loading: 'Retrieving information from map service...',
        query_empty_response: 'No response received from map service at this location. Perhaps try somewhere else or try querying another layer?',
        retrieving_error: 'Retrieving error...',
        query_layer: 'Query a certain location of this map for more information by clicking somewhere on the map',
        edit_layer: 'Edit map',
        remove_layer: 'Remove map',
        satellite_imagery: 'Display satellite imagery',
        select_map: 'Select a map to add',
        opacity_label: 'Opacity',
        opacity_label_tooltip: 'Adjust how transparant the layer is',
        transparent_label: 'Transparent',
        transparent_label_tooltip: 'Make the map background transparent',
        errors: {
          network: 'Network error',
          network_description: 'Received the following status code when retrieving information: ',
          see_console: 'See web console for more details',
          timeout: 'Timeout',
          timeout_description: 'Retrieving response took more than 10s, probably network issue',
          parse_fail: 'Could not parse error message'
        },
        update: {
          message: 'WME Open Maps has been updated! Changelog:',
          v2_3_0: 'Complete rework of the userscript\n- Display multiple maps at the same time\n- Make it possible to query map layers',
          v2_3_1: '- Fixes loading and saving of map state\n- Fixed some bugs concerning map ordering\n- Gray background added to map loading indicator\n- Adjusted BAG map queryability',
          v2_3_2: '- Fixes bug where removing a map also internally removed the maps below\n- Layer querying will only now only take place on visible queryable layers\n- Small changes to boundary and querying options in some maps',
          v2_3_3: '- Fixes to map layer reordering',
          v2_3_4: '- Small UI improvements and internal code refactoring',
          v2_3_5: '- Slightly improved map query response handling\n- Fixed TamperMonkey notices about accessing external resources'
        }
      },
      nl: {
        tab_title: 'Open Maps',
        maps_title: 'Actieve kaarten',
        no_local_maps: 'Geen lokale kaarten gevonden',
        expand: 'Klik om uit te breiden',
        collapse: 'Klik om te verbergen',
        hideshow_layer: 'Verberg/Toon kaart',
        query_window_title: 'Kaartlocatie doorzoeken',
        query_window_loading: 'Informatie aan het opvragen bij kaartdienst...',
        query_empty_response: 'Geen antwoord ontvangen van de kaartdienst op deze locatie. Misschien kan je een andere locatie proberen of een andere laag bevragen?',
        retrieving_error: 'Fout aan het ophalen...',
        query_layer: 'Doorzoek een bepaalde locatie op deze kaart voor meer informatie door ergens op de kaart te klikken',
        edit_layer: 'Pas de kaart aan',
        remove_layer: 'Verwijder kaart',
        satellite_imagery: 'Geef satellietbeelden weer',
        select_map: 'Selecteer een kaart om toe te voegen',
        opacity_label: 'Doorzichtigheid',
        opacity_label_tooltip: 'Wijzig de doorzichtigheid van de kaart',
        transparent_label: 'Transparent',
        transparent_label_tooltip: 'Maak de achtergrond van de kaart transparent',
        errors: {
          network: 'Networkfout',
          network_description: 'Bij het opvragen van informatie werd de volgende statuscode ontvangen: ',
          see_console: 'Bekijk de browserconsole voor meer informatie',
          timeout: 'Time-out',
          timeout_description: 'Antwoord verkrijgen duurde langer dan 10 seconden, waarschijnlijk netwerkprobleem',
          parse_fail: 'Kan foutmelding niet verwerken'
        },
        update: {
          message: 'Nieuwe versie van WME Open Maps geïnstalleerd! Veranderingen:',
          v2_3_0: 'Complete herwerking van het userscript\n- Geef meerdere kaarten tegelijk weer\n- Maak het mogelijk om kaarten te doorzoeken',
          v2_3_1: '- Het inladen en opslagen van de kaarten is opgelost\n- Enkele bugs geplet rond het ordenen van kaarten\n- Een grijze achtergrond toegevoegd aan de laadindicator voor kaarten\n- De doorzoakbaarheid van de BAG-kaart is aangepast',
          v2_3_2: '- Probleem opgelost waarbij het verwijderen van een kaart alle onderliggende kaarten ook verwijderde\n- Het bevragen van een kaart gebeurt nu enkel op zichtbare bevraagbare lagen\n- Kleine veranderingen aan de grenzen en bevragingsinstellingen van sommige kaarten',
          v2_3_3: '- Het verplaatsen van lagen van een kaart is hersteld',
          v2_3_4: '- Kleine veranderingen aan de UI en interne herwerking van code',
          v2_3_5: '- Licht verbeterde verwerking van kaartopzoekingen\n- Probleem opgelost met TamperMonkey-meldingen over het gebruik van externe bronnen'
        }
      },
      fr: {
        opacity_label: 'Opacité',
        tab_title: 'Open Maps',
        no_local_maps: 'Aucune carte disponible ici'
      }
    };
    om_strings.en_GB = om_strings.en;
    for (var i = 0; i < I18n.availableLocales.length; i++) {
      var locale = I18n.availableLocales[i];
      if (I18n.translations[locale]) {
        I18n.translations[locale].openmaps = cloneInto(om_strings[locale], unsafeWindow);
      }
    }
    
    // List of available maps
    var maps = {
      '3201': { id: 3201, url: 'https://geoservices.informatievlaanderen.be/raadpleegdiensten/GRB/wms', crs: 'EPSG:3857', bbox: new OL.Bounds(280525.11676, 6557859.253174342, 661237.77522, 6712007.501374752), format: 'image/png', title: 'GRB Vlaanderen', abstract: 'BE: Via de WMS GRB kan je het Grootschalig Referentiebestand (GRB) opvragen en visualiseren als een kaart. De WMS GRB omvat alle GRB-gegevens gebaseerd op het GRBgis product. De gebruiker kan selecteren welke GRB-gegevens gevisualiseerd moeten worden en in welke volgorde. Voor een gedetailleerde databeschrijving van het GRB raadpleegt u best het GRB-objectenhandboek via www.agiv.be/producten/grb/objectcatalogus/entiteiten.', attribution: 'Agentschap voor Geografische Informatie Vlaanderen', queryable: false, default_layers: [ 'GRB_BSK' ], layers: { 'GRB_BSK': { queryable: false, title: 'GRB-basiskaart', abstract: 'Deze laag omvat alle (GRB-) entiteiten die zichtbaar zijn in de GRB-basiskaart' } } },
      '3202': { id: 3202, url: 'https://geoservices.informatievlaanderen.be/raadpleegdiensten/omwrgbmrvl/wms', crs: 'EPSG:3857', bbox: new OL.Bounds(280525.11676, 6557859.253174342, 661237.77522, 6712007.501374752), format: 'image/jpeg', title: 'Orthomozaïek Vlaanderen', abstract: 'BE: WMS die de compilatie weergeeft van de meest recente middenschalige orthofotomozaïeken uit de wintervluchten die voor ieder deel van Vlaanderen beschikbaar zijn die wordt bijgewerkt telkens er een nieuw deel beschikbaar is.', attribution: 'Agentschap voor Geografische Informatie Vlaanderen', queryable: true, default_layers: [ 'Ortho', 'Vliegdagcontour' ], layers: { 'Ortho': { queryable: false, title: 'Orthofotomozaïek, middenschalig, winteropnamen, kleur, meest recent, Vlaanderen', abstract: 'Deze rasterlaag is een compilatie van de meest recente orthofotomozaëken (winteropnamen) die voor ieder deel  van Vlaanderen beschikbaar zijn en wordt  bijgewerkt telkens er een nieuw deel ingewonnen is. De compilatie heeft een grondresolutie van 25 cm.' }, 'Vliegdagcontour': { queryable: true, title: 'Vliegdagcontour Orthofotomozaïek', abstract: 'Deze vectorlaag geeft voor ieder deel van de rastercompilatie de opnamedatum weer.' } } },
      '3203': { id: 3203, url: 'https://geoservices.wallonie.be/arcgis/services/TOPOGRAPHIE/PICC_VDIFF/MapServer/WMSServer', crs: 'EPSG:3857', bbox: new OL.Bounds(295477.314255, 740430.033845, 6347477.319654, 6640885.073618), format: 'image/png', title: 'PICC, Service de visualisation', abstract: 'BE: Service de visualisation du Projet Informatique de Cartographie Continue (PICC)', attribution: 'Service public de Wallonie', queryable: true, default_layers: [ '3', '4', '6', '8', '9', '17', '18', '19', '25', '26', '27' ], layers: { '1': { queryable: true, title: 'Relief: ligne' }, '3': { queryable: true, title: 'Hydrographie: bord' }, '4': { queryable: true, title: 'Hydrographie: axe' }, '6': { queryable: true, title: 'Reseau ferroviaire: ligne' }, '8': { queryable: true, title: 'Voirie: axe >= 5k' }, '9': { queryable: true, title: 'Voirie: axe' }, '10': { queryable: true, title: 'Voirie: ligne' }, '12': { queryable: true, title: 'Occupation du sol: surface' }, '13': { queryable: true, title: 'Occupation du sol: bord' }, '14': { queryable: true, title: 'Occupation du sol: ligne' }, '15': { queryable: true, title: 'Occupation du sol: point' }, '17': { queryable: true, title: 'Construction: emprise du batiment' }, '18': { queryable: true, title: 'Construction: ouvrage d\'art: bord' }, '19': { queryable: true, title: 'Construction: bord du batiment' }, '21': { queryable: true, title: 'Equipement: surface' }, '22': { queryable: true, title: 'Equipement: axe' }, '23': { queryable: true, title: 'Equipement: ligne' }, '24': { queryable: true, title: 'Equipement: point' }, '25': { queryable: true, title: 'Symbologie' }, '26': { queryable: true, title: 'Adresses' }, '27': { queryable: true, title: 'Toponymie' } } },
      '3204': { id: 3204, url: 'http://geoserver.gis.irisnet.be/geoserver/wms', crs: 'EPSG:31370', bbox: new OL.Bounds(471578, 6579050, 499555, 6606337), format: 'image/png', title: 'CIRB', abstract: 'BE: Web Map Service for the CIRB layers', attribution: 'Irisnet GIS', queryable: true, default_layers: [ 'urbisNL' ], layers: { 'urbisNL': { queryable: false, title: 'Urbis Base Map NL', abstract: 'This layer represents the base map in dutch.' }, 'urbisFR': { queryable: false, title: 'Urbis Base Map FR', abstract: 'This layer represents the base map in french.' }, 'urbis:ortho2014': { queryable: false, title: 'Ortho 2014', abstract: 'This layer is ortho 2014 in the Brussels region' }, 'urbis:ortho2012': { queryable: false, title: 'Ortho 2012', abstract: 'This layer is ortho 2012 in the Brussels region' }, 'urbis:ortho2009': { queryable: false, title: 'Ortho 2009', abstract: 'This layer is ortho 2009 in the Brussels region' }, 'urbis:ortho2004': { queryable: false, title: 'Ortho 2004', abstract: 'This layer is ortho 2004 in the Brussels region' }, 'urbisFRGray': { queryable: false, title: 'Urbis Base Map Gray FR', abstract: 'This layer represents the gray base map in french.' }, 'urbisNLGray': { queryable: false, title: 'Urbis Base Map Gray NL', abstract: 'This layer represents the gray base map in dutch.' }, 'urbis:LabeledStreetAxe': { queryable: false, title: 'Labeled Street Axe', abstract: 'Labeled StreetAxe for OSIRIS, bug fix for the juxtaposition of street name on building' }, 'urbis:URB_A_ADPT': { queryable: false, title: 'Address points', abstract: 'This layer is the localization of address points of the Brussels Region' }, 'urbis:URB_A_BU': { queryable: true, title: 'Buildings', abstract: 'This layer represents the buildings of the Brussels Region' }, 'urbis:URB_A_MD': { queryable: true, title: 'Monitoring districts', abstract: 'This layer reprensent the monitoring districts of the Brussels Region' }, 'urbis:URB_A_MU': { queryable: true, title: 'Municipalities', abstract: 'This layer represents the municipalities of the Brussels Region' }, 'urbis:URB_A_MY_SA': { queryable: false, title: 'Street axes', abstract: 'This layer represents the axes of the street of the Brussels Region' }, 'urbis:URB_A_MY_SS': { queryable: false, title: 'Street sections', abstract: 'This layer represents the street sections of the Brussels Region' }, 'urbis:URB_A_MZ': { queryable: true, title: 'Municipal zips', abstract: 'This layer is the zip of the municipality of the Brussels Region' }, 'urbis:URB_A_POL': { queryable: true, title: 'Police districts', abstract: 'This layer is the police districts of the Brussels Region' }, 'urbis:URB_A_RE': { queryable: false, title: 'Region', abstract: 'This layer is the Brussels Region' }, 'urbis:URB_A_SD': { queryable: true, title: 'Statistical districts', abstract: 'This layer represents the limit of the statistical districts of the Brussels Region' }, 'urbis:URB_A_SN': { queryable: false, title: 'Street nodes', abstract: 'This layer represents the street nodes. Each node is an intersection or an extremity of a street axe' }, 'urbis:URB_M_RTLINE': { queryable: false, title: 'Rail tracks', abstract: 'This layer represents the rails tracks.' }, 'urbis:URB_M_SHAPE': { queryable: true, title: 'UrbisMap shapes', abstract: 'This layer represents the shapes of UrbisMap.' }, 'urbis:URB_M_TONAME_LIN': { queryable: true, title: 'Toponymy', abstract: 'This layer represents the toponymy of public places.' }, 'urbis:URB_M_ZIPOINT': { queryable: true, title: 'Points of interest', abstract: 'This layer represents the point of zone of interest.' }, 'urbis:URB_T_LINE': { queryable: false, title: 'Urbis Topo Lines', abstract: 'This layer represents the topo lines.' }, 'urbis:URB_T_POINT': { queryable: true, title: 'Urbis Topo Points', abstract: 'This layer represents the topo points.' }, 'urbis:URB_A_SI_POINT_VW': { queryable: false, title: 'Street sides' }, 'urbis:MuNeighbour': { queryable: true, title: 'Neighbour Municipalities' }, 'urbis:Highways': { queryable: false, title: 'Highways' } } },
      '3205': { id: 3205, url: 'https://geo.agiv.be/ogc/wms/gipodpubliek', crs: 'EPSG:3857', bbox: new OL.Bounds(280525, 6557859, 661237, 6712007), format: 'image/png', title: 'GIPOD Publieke Informatie', abstract: 'BE: Deze WMS geeft een overzicht van alle concreet geplande en in uitvoering zijnde werken, manifestaties en andere innames op het openbaar domein met hun bijhorende omleidingen en verwachte hinder, voor de komende maand.', attribution: 'Agentschap voor Geografische Informatie Vlaanderen', queryable: true, default_layers: [ 'ManOml', 'ManCon', 'ManIcoon', 'WoOml', 'WoCon', 'WoIcoon' ], layers: { 'ManOml': { queryable: false, title: 'Omleidingen van de manifestaties', abstract: 'Deze laag geeft een overzicht van alle omleidingen, horend bij manifestaties en andere innames op het openbaar domein, voor de komende maand.' }, 'ManCon': { queryable: true, title: 'Manifestaties contour', abstract: 'Deze laag geeft een overzicht met contouren van alle manifestaties en andere innames op het openbaar domein en de verwachte hinder voor de komende maand.' }, 'ManIcoon': { queryable: false, title: 'Manifestaties icoon', abstract: 'Deze laag geeft een overzicht met iconen van alle manifestaties en andere innames op het openbaar domein en de verwachte hinder voor de komende maand.' }, 'WoOml': { queryable: false, title: 'Omleidingen van de werkopdrachten', abstract: 'Deze laag geeft een overzicht van alle omleidingen, horend bij werkopdrachten op het openbaar domein, voor de komende maand.' }, 'WoCon': { queryable: true, title: 'Werkopdrachten contour', abstract: 'Deze laag geeft een overzicht met contouren van alle werkopdrachten op het openbaar domein en de verwachte hinder voor de komende maand.' }, 'WoIcoon': { queryable: false, title: 'Werkopdrachten icoon', abstract: 'Deze laag geeft een overzicht met iconen van alle werkopdrachten op het openbaar domein en de verwachte hinder voor de komende maand.' } } },
      '3206': { id: 3206, url: 'https://geoservices.informatievlaanderen.be/raadpleegdiensten/omw/wms', crs: 'EPSG:3857', bbox: new OL.Bounds(280525.11676, 6557859.253174342, 661237.77522, 6712007.501374752), format: 'image/jpeg', title: 'Orthomozaïek Vl. Tijdsreeksen', abstract: 'BE: WMS met de tijdsreeks van middenschalige orthofotomozaïeken met een resolutie van 25cm, gebiedsdekkend voor Vlaanderen', attribution: 'Agentschap voor Geografische Informatie Vlaanderen', queryable: true, default_layers: [ 'OMWRGB15VL', 'OMWRGB15VL_vdc' ], layers: { 'OMWRGB15VL': { queryable: false, title: 'Winteropnamen, 2015', abstract: 'Deze rasterlaag is een compilatie van de orthofotomozaïeken (winteropnamen) die voor Vlaanderen in 2015 werden aangemaakt. De compilatie heeft een grondresolutie van 25cm.' }, 'OMWRGB15VL_vdc': { queryable: true, title: 'Winteropnamen, 2015, vliegdagcontour', abstract: 'Vectorlaag die voor ieder deel van het bijhorende product de opnamedatum weergeeft.' }, 'OMWRGB14VL': { queryable: false, title: 'Winteropnamen, 2014', abstract: 'Deze rasterlaag is een compilatie van de orthofotomozaïeken (winteropnamen) die voor Vlaanderen in 2014 werden aangemaakt. De compilatie heeft een grondresolutie van 25cm.' }, 'OMWRGB14VL_vdc': { queryable: true, title: 'Winteropnamen, 2014, vliegdagcontour', abstract: 'Vectorlaag die voor ieder deel van het bijhorende product de opnamedatum weergeeft.' }, 'OMWRGB13VL': { queryable: false, title: 'Winteropnamen, 2013', abstract: 'Deze rasterlaag is een compilatie van de orthofotomozaïeken (winteropnamen) die voor Vlaanderen in 2013 werden aangemaakt. De compilatie heeft een grondresolutie van 25cm.' }, 'OMWRGB13VL_vdc': { queryable: true, title: 'Winteropnamen, 2013, vliegdagcontour', abstract: 'Vectorlaag die voor ieder deel van het bijhorende product de opnamedatum weergeeft.' }, 'OMWRGB12VL': { queryable: false, title: 'Winteropnamen, 2012', abstract: 'Deze rasterlaag is een compilatie van de orthofotomozaïeken (winteropnamen) die voor Vlaanderen in 2012 werden aangemaakt. De compilatie heeft een grondresolutie van 25cm.' }, 'OMWRGB12VL_vdc': { queryable: true, title: 'Winteropnamen, 2012, vliegdagcontour', abstract: 'Vectorlaag die voor ieder deel van het bijhorende product de opnamedatum weergeeft.' }, 'OMWRGB08_11VL': { queryable: false, title: 'Winteropnamen, 2008-2011', abstract: 'Deze rasterlaag is een compilatie van de orthofotomozaïeken (winteropnamen) die voor Vlaanderen in de periode 2008-2011 werden aangemaakt. De compilatie heeft een grondresolutie van 25cm.' }, 'OMWRGB08_11VL_vdc': { queryable: true, title: 'Winteropnamen, 2008-2011, vliegdagcontour', abstract: 'Vectorlaag die voor ieder deel van het bijhorende product de opnamedatum weergeeft.' }, 'OMWRGB05_07VL': { queryable: false, title: 'Winteropnamen, 2005-2007', abstract: 'Deze rasterlaag is een compilatie van de orthofotomozaïeken (winteropnamen) die voor Vlaanderen in de periode 2005-2007 werden aangemaakt. De compilatie heeft een grondresolutie van 25cm.' }, 'OMWRGB05_07VL_vdc': { queryable: true, title: 'Winteropnamen, 2005-2007, vliegdagcontour', abstract: 'Vectorlaag die voor ieder deel van het bijhorende product de opnamedatum weergeeft.' }, 'OMWRGB00_03VL': { queryable: false, title: 'Winteropnamen, 2000-2003', abstract: 'Deze rasterlaag is een compilatie van de orthofotomozaïeken (winteropnamen) die voor Vlaanderen in de periode 2000-2003 werden aangemaakt. De compilatie heeft een grondresolutie van 25cm.' }, 'OMWRGB00_03VL_vdc': { queryable: true, title: 'Winteropnamen, 2000-2003, vliegdagcontour', abstract: 'Vectorlaag die voor ieder deel van het bijhorende product de opnamedatum weergeeft.' } } },
      '3101': { id: 3101, url: 'https://geodata.nationaalgeoregister.nl/bag/wms', crs: 'EPSG:3857', bbox: new OL.Bounds(368979.10475029063, 6575606.8651319705, 785145.4377025769, 7057109.006193211), format: 'image/png', title: 'BAG', abstract: 'NL: De gegevens bestaan uit BAG-panden en een deelselectie van BAG-gegevens van deze panden en de zich daarin bevindende verblijfsobjecten. Ook de ligplaatsen en standplaatsen zijn hierin opgenomen met een deelselectie van BAG-gegevens. De gegevens van de nummeraanduiding zijn in deze services onderdeel van de adresseerbare objecten, hierbij wordt slechts 1 adres opgenomen, dus objecten met meerdere adressen (hoofd- en nevenadressen) zijn niet compleet. In deze services zitten dus niet alle BAG adressen. Wij adviseren u, aangezien er sprake is van beperkte gegevens, om in de webservice BAG Bevragen de actuele gegevens te controleren. Dit kan ook in een van de andere BAG producten: BAG Web, BAG Extract of BAG Compact. BAG Bevragen: http://www.kadaster.nl/web/artikel/productartikel/BAG-Bevragen.htm Andere BAG producten: http://www.kadaster.nl/web/Themas/Registraties/BAG/BAGartikelen/BAG-producten.htm De service wordt dagelijks geactualiseerd.', attribution: 'BAG', queryable: true, default_layers: [ 'bag','ligplaats','pand','verblijfsobject','woonplaats','standplaats' ], layers: { 'bag': { queryable: false, title: 'bag' }, 'ligplaats': { queryable: false, title: 'ligplaats' }, 'pand': { queryable: false, title: 'pand' }, 'standplaats': { queryable: false, title: 'standplaats' }, 'verblijfsobject': { queryable: true, title: 'verblijfsobject' }, 'woonplaats': { queryable: false, title: 'woonplaats' } } },
      '3102': { id: 3102, url: 'http://webservices.gbo-provincies.nl/lufo/services/wms', crs: 'EPSG:28992', bbox: new OL.Bounds(332126.70701065, 6575606.8651319705, 837006.18392751, 7070731.52499591), format: 'image/jpeg', title: 'Luchtfoto (GBO provincies)', abstract: 'NL: Luchtfoto wms door GBO provincies', attribution: 'GBO provincies', queryable: false, default_layers: [ 'actueel_winter' ], layers: { 'actueel_winter': { queryable: false, title: 'Luchtfoto 2014 (winter)' }, 'actueel_zomer': { queryable: false, title: 'Luchtfoto 2014 (zomer)' }, 'luchtfoto_2013_winter': { queryable: false, title: 'Luchtfoto 2013 (winter)' }, 'luchtfoto_2013_zomer': { queryable: false, title: 'Luchtfoto 2013 (zomer)' }, 'luchtfoto_2012_winter': { queryable: false, title: 'Luchtfoto 2012 (winter)' }, 'luchtfoto_2012_zomer': { queryable: false, title: 'Luchtfoto 2012 (zomer)' }, 'luchtfoto_2011': { queryable: false, title: 'Luchtfoto 2011' }, 'luchtfoto_2010': { queryable: false, title: 'Luchtfoto 2010' }, 'luchtfoto_2009': { queryable: false, title: 'Luchtfoto 2009' }, 'luchtfoto_2008': { queryable: false, title: 'Luchtfoto 2008' } } },
      '3103': { id: 3103, url: 'https://geodata.nationaalgeoregister.nl/weggeg/wms', crs: 'EPSG:3857', bbox: new OL.Bounds(385276.75763551984, 6575606.8651319705, 805841.7938525073, 7065365.041596532), format: 'image/png', title: 'Weggegevens WMS', abstract: 'NL: De service van Weggegevens bevat op dit moment de lagen maximum snelheden en rijstroken van de rijkswegen.', attribution: 'PDOK', queryable: true, default_layers: [ 'weggegaantalrijbanen','weggegmaximumsnelheden' ], layers: { 'weggegaantalrijbanen': { queryable: true, title: 'Weggegevens aantal rijbanen', }, 'weggegmaximumsnelheden': { queryable: true, title: 'Weggegevens maximumsnelheden', } } },
      '3104': { id: 3104, url: 'https://geodata.nationaalgeoregister.nl/bgt/wms', crs: 'EPSG:28992', bbox: new OL.Bounds(333958.4723798207, 6575606.8651319705, 779236.435552915, 6982997.920389788), format: 'image/png', title: 'INSPIRE View Service PDOK', abstract: 'NL: De INSPIRE View Service PDOK is een overkoepelende View Service die voor de volgende datasets INSPIRE datasets ontsluit: Hydrografie, Vervoersnetwerken, Beschermde gebieden, Geografische namen', attribution: 'PDOK', queryable: true, default_layers: [ 'bgtstandaard' ], layers: { 'bgtachtergrond': { queryable: true, title: 'BGT achtergrond' }, 'bgtstandaard': { queryable: true, title: 'BGT standaard' }, 'bgt_v_nummeraanduiding': { queryable: true, title: 'Nummeraanduidingen' }, 'bgtvulling': { queryable: true, title: 'BGT vulling' } } }
    };
    
    checkVersion();

    // List of map handles
    var handles = [];
    var tab = addOpenMapsTab();

    // Satellite imagery toggle
    var satImageryDiv = document.createElement('div');
    satImageryDiv.className = 'controls-container';
    var satImagery = document.createElement('input');
    satImagery.type = 'checkbox';
    satImagery.id = 'satImagery-on';
    satImagery.checked = Waze.map.layers[0].getVisibility();
    satImagery.addEventListener('click', function() {
      Waze.map.layers[0].setVisibility(this.checked);
    });
    Waze.map.layers[0].events.register('visibilitychanged', null, exportFunction(function() {
      satImagery.checked = Waze.map.layers[0].getVisibility();
    }, unsafeWindow));
    satImageryDiv.appendChild(satImagery);
    var satImageryLabel = document.createElement('label');
    satImageryLabel.htmlFor = 'satImagery-on';
    satImageryLabel.appendChild(document.createTextNode(I18n.t('openmaps.satellite_imagery')));
    satImageryDiv.appendChild(satImageryLabel);
    tab.appendChild(satImageryDiv);
    
    // List of maps visible in Open Maps
    var title = document.createElement('h4');
    title.appendChild(document.createTextNode(I18n.t('openmaps.maps_title')));
    title.style.marginBottom = '5px';
    tab.appendChild(title);
    var handleList = document.createElement('ul');
    $(handleList).sortable(cloneInto({
      forcePlaceholderSize: true,
      placeholderClass: 'result',
      handle: '.title'
    }, unsafeWindow)).bind('sortupdate', exportFunction(function(e, ui) {
      var movedHandle = handles.splice(ui.oldElementIndex, 1)[0];
      handles.splice(ui.elementIndex, 0, movedHandle);
      if (ui.elementIndex >= 0 && ui.elementIndex < handles.length) { // sanity check
        var aerialImageryIndex = Waze.map.getLayerIndex(Waze.map.getLayersBy('uniqueName', 'satellite_imagery')[0]);
        Waze.map.setLayerIndex(movedHandle.layer, (aerialImageryIndex >= 0 ? aerialImageryIndex : 0) + ui.elementIndex + 1);
      }
      saveMapState();
    }, unsafeWindow));
    handleList.className = 'result-list';
    tab.appendChild(handleList);

    // Select box to add new Open Maps maps
    var addMap = document.createElement('select');
    addMap.className = 'form-control';
    addMap.style.margin = '8px 0';
    updateMapSelector();
    Waze.map.events.register("moveend", null, exportFunction(updateMapSelector, unsafeWindow));
    addMap.addEventListener('change', function() {
      if (addMap.selectedIndex != 0) {
        var mapId = addMap.options[addMap.selectedIndex].value;
        handles.push(new MapHandle(maps[mapId]));
        saveMapState();
        addMap.selectedIndex = 0;
      }
    });
    tab.appendChild(addMap);

    var footer = document.createElement('p');
    try {
      footer.appendChild(document.createTextNode(GM_info.script.name + ': v' + GM_info.script.version));
    } catch (e) {
      // Probably no support for GM_info, ignore
    }
    footer.style.fontSize = '11px';
    tab.appendChild(footer);
    
    var idGenerator = (function() {
      var counter = 0;
      return {
        getNext: function() {
          return counter++;
        }
      };
    })();
    
    // Try to convert any previous localStorage data
    if (typeof localStorage.OM_previousMap != 'undefined' || typeof localStorage.OM_opacity != 'undefined') {
      if (typeof maps[localStorage.OM_previousMap] != 'undefined') {
        handles.push(new MapHandle(maps[localStorage.OM_previousMap], { opacity: localStorage.OM_opacity }));
        saveMapState();
      }
      localStorage.removeItem('OM_previousMap');
      localStorage.removeItem('OM_opacity');
    }
    // Reload previous map(s)
    if (typeof localStorage.OpenMaps != 'undefined') {
      var storage = JSON.parse(localStorage.OpenMaps);
      storage.state.active.forEach(function(mapHandle, i) {
        if (maps[mapHandle.mapId] == undefined) { // no strict equal as null should fail as well
          storage.state.active.splice(i, 1);
          localStorage.OpenMaps = JSON.stringify(storage);
          return;
        }
        handles.push(new MapHandle(maps[mapHandle.mapId], {
          opacity: mapHandle.opacity,
          layers: mapHandle.layers,
          hidden: mapHandle.hidden,
          transparent: mapHandle.transparent
        }));
        saveMapState();
      });
    }
    // Add the control to catch a click on the map area for retrieving map information
    var queryParams = null;
    var queryWindow = document.createElement('div');
    queryWindow.style.display = 'none';
    queryWindow.style.top = '10px';
    queryWindow.style.left = '100px';
    queryWindow.style.right = '60px';
    queryWindow.style.backgroundColor = '#fff';
    queryWindow.style.border = '2px solid #ddd';
    queryWindow.style.padding = '5px';
    queryWindow.style.color = '#000';
    queryWindow.style.zIndex = '2000';
    queryWindow.style.position = 'absolute';
    var queryWindowClose = document.createElement('span');
    queryWindowClose.appendChild(document.createTextNode(''));
    queryWindowClose.style.fontFamily = 'FontAwesome';
    queryWindowClose.style.float = 'right';
    queryWindowClose.style.cursor = 'pointer';
    queryWindowClose.style.fontSize = '20px';
    queryWindowClose.style.color = '#fff';
    queryWindowClose.style.padding = '4px';
    queryWindowClose.style.backgroundColor = '#000';
    queryWindowClose.addEventListener('click', exportFunction(function() {
      queryWindow.style.display = 'none';
    }, unsafeWindow));
    queryWindow.appendChild(queryWindowClose);
    var queryWindowTitle = document.createElement('h2');
    queryWindowTitle.style.textAlign = 'center';
    queryWindowTitle.style.fontWeight = 'bold';
    queryWindowTitle.appendChild(document.createTextNode(I18n.t('openmaps.query_window_title')));
    queryWindow.appendChild(queryWindowTitle);
    var queryWindowLoading = document.createElement('p');
    queryWindowLoading.style.textAlign = 'center';
    queryWindowLoading.style.fontSize = '21px';
    var queryWindowLoadingSpinner = document.createElement('span');
    queryWindowLoadingSpinner.className = 'fa fa-spinner fa-pulse';
    queryWindowLoading.appendChild(queryWindowLoadingSpinner);
    queryWindowLoading.appendChild(document.createTextNode(' ' + I18n.t('openmaps.query_window_loading')));
    queryWindow.appendChild(queryWindowLoading);
    var queryWindowContent = document.createElement('div');
    queryWindowContent.style.overflow = 'auto';
    queryWindowContent.style.fontSize = '14px';
    queryWindow.appendChild(queryWindowContent);
    document.getElementById('WazeMap').appendChild(queryWindow);
    var querySymbol = document.createElement('span');
    querySymbol.appendChild(document.createTextNode(''));
    querySymbol.style.fontFamily = 'FontAwesome';
    querySymbol.style.fontSize = '42px';
    querySymbol.style.float = 'left';
    querySymbol.style.margin = '0 15px 30px';
    var getFeatureInfoControl = new OL.Control(cloneInto({
      id: 'GetFeatureInfoControl'
    }, unsafeWindow));
    Waze.map.addControl(getFeatureInfoControl);
    var clickHandler = new OL.Handler.Click(getFeatureInfoControl, cloneInto({
      'click': function(e) {
        getFeatureInfoControl.deactivate();
        queryParams.callback();
        var queryUrl = queryParams.url + '?SERVICE=WMS&REQUEST=GetFeatureInfo&BBOX=' + Waze.map.getExtent().toBBOX() +
            '&LAYERS=' + queryParams.layers + '&QUERY_LAYERS=' + queryParams.layers +
            '&HEIGHT=' + Waze.map.getSize().h + '&WIDTH=' + Waze.map.getSize().w +
            '&VERSION=1.3.0&CRS=EPSG:3857&I=' + e.xy.x + '&J=' + e.xy.y + '&INFO_FORMAT=text/html';
        queryParams = null;
        queryWindowLoading.style.display = 'block';
        while (queryWindowContent.firstChild) {
          queryWindowContent.removeChild(queryWindowContent.firstChild);
        }
        queryWindow.style.display = 'block';
        GM_xmlhttpRequest({
          method: 'GET',
          headers: {
            Accept: 'text/xml'
          },
          url: queryUrl,
          timeout: 10000,
          onload: function(response) {
            queryWindowLoading.style.display = 'none';
            if (response.status == 200) {
              if (!response.responseXML) {
                response.responseXML = new DOMParser().parseFromString(response.responseText, "text/xml");
              }
              // While probably not 100% waterproof, at least this should counter most XSS vectors
              var unwantedNodes = response.responseXML.querySelectorAll('javascript,iframe,frameset,applet,embed,object,style');
              for (var i = 0; i < unwantedNodes.length; i++) {
                unwantedNodes[i].parentNode.removeChild(unwantedNodes[i]);
              }
              var body = response.responseXML.querySelector('body');
              var content = body.textContent.trim();
              if (body && content.length != 0) {
                removeUnsafeAttributes(body);
                queryWindowContent.innerHTML = body.innerHTML;
                queryWindow.style.display = 'block';
              } else {
                querySymbol.style.color = '#999';
                queryWindowContent.appendChild(querySymbol);
                var emptyResponse = document.createElement('p');
                emptyResponse.appendChild(document.createTextNode(I18n.t('openmaps.query_empty_response')));
                queryWindowContent.appendChild(emptyResponse);
              }
            } else {
              querySymbol.style.color = 'red';
              queryWindowContent.appendChild(querySymbol);
              var errorResponseTitle = document.createElement('p');
              errorResponseTitle.appendChild(document.createTextNode(I18n.t('openmaps.errors.network')));
              queryWindowContent.appendChild(errorResponseTitle);
              var errorResponse = document.createElement('p');
              errorResponse.appendChild(document.createTextNode(I18n.t('openmaps.errors.network_description') + response.statusText));
              queryWindowContent.appendChild(errorResponse);
            }
          },
          ontimeout: function(e) {
            log(e);
            queryWindowLoading.style.display = 'none';
            querySymbol.style.color = 'orange';
            queryWindowContent.appendChild(querySymbol);
            var timeoutResponse = document.createElement('p');
            timeoutResponse.appendChild(document.createTextNode(I18n.t('openmaps.errors.timeout_description')));
            queryWindowContent.appendChild(timeoutResponse);
          },
          onerror: function(e) {
            log(e);
            queryWindowLoading.style.display = 'none';
            querySymbol.style.color = 'red';
            queryWindowContent.appendChild(querySymbol);
            var errorResponseTitle = document.createElement('p');
            errorResponseTitle.appendChild(document.createTextNode(I18n.t('openmaps.errors.network')));
            queryWindowContent.appendChild(errorResponseTitle);
            var errorResponse = document.createElement('p');
            errorResponse.appendChild(document.createTextNode(I18n.t('openmaps.errors.see_console')));
            queryWindowContent.appendChild(errorResponse);
          }
        });
      }
    }, unsafeWindow, { cloneFunctions: true }));
    getFeatureInfoControl.handler = clickHandler;
    
    function removeUnsafeAttributes(node) {
      if (node.nodeType == Node.ELEMENT_NODE) {
        for (var i = 0; i < node.attributes.length; i++) {
          var attrName = node.attributes[i].name.toLowerCase();
          if (attrName.startsWith('on') || attrName == 'style' || attrName == 'class' || (attrName == 'href' && node.attributes[i].value.trim().toLowerCase().startsWith('javascript:'))) {
            node.removeAttribute(attrName);
          }
        }
        if (node.localName == 'table') {
          node.setAttribute('border', '1');
        }
      }
      for (var i = 0; i < node.childNodes.length; i++) {
        removeUnsafeAttributes(node.childNodes[i]);
      }
    }
    
    function checkVersion() {
      var version = localStorage.OpenMaps_version,
          scriptVersion = GM_info.script.version;
      if (!version) {
        localStorage.OpenMaps_version = scriptVersion;
      } else if (version !== scriptVersion) {
        if (versions.indexOf(version) === -1) {
          // There's tampering happening if we arrive here, just set to current version and ignore issue
          localStorage.OpenMaps_version = scriptVersion;
          return;
        }
        var message = I18n.t('openmaps.update.message');
        for (var i = versions.indexOf(version)+1; i < versions.length; i++) {
          message += '\nv' + versions[i] + ':\n' + I18n.t('openmaps.update.v' + versions[i].replace(/\./g, '_'));
        }
        localStorage.OpenMaps_version = scriptVersion;
        alert(message);
      }
    }
    
    function updateMapSelector() {
      var localMaps = [];
      Object.keys(maps).forEach(function(id) {
        if (maps[id].bbox.intersectsBounds(Waze.map.getExtent())) {
          localMaps.push(maps[id]);
        }
      });
      while (addMap.firstChild) {
        addMap.removeChild(addMap.firstChild);
      }
      if (localMaps.length == 0) {
        var noMaps = document.createElement('option');
        noMaps.text = I18n.t('openmaps.no_local_maps');
        addMap.appendChild(noMaps);
      } else {
        localMaps.sort(function(a, b) {
          return a.title.localeCompare(b.title);
        })
        var selectMap = document.createElement('option');
        selectMap.text = I18n.t('openmaps.select_map');
        addMap.appendChild(selectMap);
        localMaps.forEach(function(map) {
          var option = document.createElement('option');
          option.text = map.title;
          option.value = map.id;
          option.title = map.abstract;
          addMap.appendChild(option);
        });
      }
      // Have some active maps moved out of view?
      handles.forEach(function(handle) {
        var handleIsLocal = localMaps.find(function(map) { return map.id == handle.mapId }) != undefined;
        if (handleIsLocal && handle.outOfArea) {
          handle.outOfArea = false;
          handle.layer.setVisibility(!handle.hidden);
          handle.updateVisibility();
        }
        if (!handleIsLocal && !handle.outOfArea) {
          handle.outOfArea = true;
          handle.layer.setVisibility(false);
          handle.updateVisibility();
        }
      });
    }

    function addOpenMapsTab() {
      var userInfo = document.getElementById('user-info'),
          tabHandles = userInfo.querySelector('.nav-tabs'),
          tabs = userInfo.querySelector('.tab-content'),
          tabHandle = document.createElement('li'),
          tab = document.createElement('div');
      tabHandle.innerHTML = '<a href="#sidepanel-openMaps" data-toggle="tab" title="' + I18n.t('openmaps.tab_title') + '"><span class="fa"></span></a>';
      tab.id = 'sidepanel-openMaps';
      tab.className = 'tab-pane';
      tabHandles.appendChild(tabHandle);
      tabs.appendChild(tab);
      return tab;
    }

    function loadTileError(url, callback) {
      // Request the tile again to check the error (OpenLayers doesn't provide this information)
      GM_xmlhttpRequest({
        method: 'GET',
        url: url,
        timeout: 10000,
        onload: function (response) {
          if (response.status == 200) {
            if (!response.responseXML) {
              response.responseXML = new DOMParser().parseFromString(response.responseText, 'text/xml');
            }
            var serviceException = response.responseXML.querySelector('ServiceException');
            if (serviceException) {
              callback({
                title: serviceException.getAttribute('code').replace(/([a-z])([A-Z])/g, '$1 $2'), // de-camelCase
                description: serviceException.textContent.trim()
              });
            } else {
              log('Failed to parse response');
              log(response);
              callback({
                title: I18n.t('openmaps.errors.parse_fail'),
                description: I18n.t('openmaps.errors.see_console')
              })
            }
          } else {
            callback({
              title: I18n.t('openmaps.errors.network'),
              description: I18n.t('openmaps.errors.network_description') + response.statusText
            });
          }
        },
        ontimeout: function(e) {
          callback({
            title: I18n.t('openmaps.errors.timeout'),
            description: I18n.t('openmaps.errors.timeout_description')
          });
        },
        onerror: function(e) {
          log('Network error');
          log(e);
          callback({
            title: I18n.t('openmaps.errors.network'),
            description: I18n.t('openmaps.errors.see_console')
          });
        }
      });
    }
      
    function saveMapState() {
      var storage;
      if (localStorage.OpenMaps) {
        storage = JSON.parse(localStorage.OpenMaps);
      }
      if (!storage) {
        storage = {};
      }
      if (!storage.state) {
        storage.state = {};
      }
      storage.state.active = [];
      handles.forEach(function(handle) {
        var handleState = {
          mapId: handle.mapId,
          opacity: handle.opacity,
          layers: handle.mapLayers,
          hidden: handle.hidden,
          transparent: handle.transparent
        };
        storage.state.active.push(handleState);
      });
      localStorage.OpenMaps = JSON.stringify(storage);
    }

    function MapHandle(map, options) {
      var self = this;
      this.layer = null;
      this.mapId = map.id;
      this.mapLayers = [];
      this.opacity = (options && options.opacity ? options.opacity : "100");
      this.outOfArea = !map.bbox.intersectsBounds(Waze.map.getExtent());
      this.hidden = (options && options.hidden ? true : false);
      this.transparent = (options && !options.transparent || map.format == 'image/jpeg' ? false : true);
      var container = document.createElement('li');
      var layerContainer = document.createElement('ul');
      var title = document.createElement('p');
      var description = document.createElement('p');
      var editContainer = document.createElement('div');
      var remove = document.createElement('span');
      var visibility = document.createElement('span');
      var query = document.createElement('span');
      var edit = document.createElement('span');
      var error = document.createElement('span');
      var loadedTiles = 0;
      var totalTiles = 0;
      var layerRedrawNeeded = false; // flag to set when a layer was made visibile/invisible
      // Deal with layers within map
      var layerKeys = Object.keys(map.layers);
      if (options && options.layers) {
        options.layers.forEach(function(oldLayer) { // Necessary if the map no longer contains certain layers that were still stored
          if (layerKeys.indexOf(oldLayer.name) != -1) {
            self.mapLayers.push(oldLayer);
            // Remove layer from map layers
            layerKeys.splice(layerKeys.indexOf(oldLayer.name), 1);
          }
        });
        layerKeys.forEach(function(layerKey) { // Add any new layers at the end of the checkboxes
          self.mapLayers.push({
            name: layerKey,
            visible: false
          });
        });
      } else { // Nothing found, apply its default layer(s)
        layerKeys.forEach(function(layerKey) {
          self.mapLayers.push({
            name: layerKey,
            visible: (map.default_layers.indexOf(layerKey) != -1)
          });
        });
      }
      
      function updateTileLoader() {
        if (loadedTiles == totalTiles) {
          loadedTiles = 0;
          totalTiles = 0;
          title.style.borderImage = 'none';
        } else {
          var percentage = parseInt(loadedTiles * 100 / totalTiles, 10);
          title.style.borderImage = 'linear-gradient(to right, #4f4, #4f4 ' + percentage + '%, #ddd ' + percentage + '%, #ddd) 100 4 stretch';
        }
      }
      
      function createIconButton(icon, title) {
        var button = document.createElement('button');
        button.style.fontFamily = 'FontAwesome';
        button.style.border = 'none';
        button.style.background = 'none';
        button.style.padding = '0 3px';
        button.style.float = 'right';
        button.style.cursor = 'pointer';
        button.style.height = 'auto';
        button.style.outline = 'none';
        button.appendChild(document.createTextNode(icon));
        if (title) {
          button.title = title;
          $(button).tooltip();
        }
        return button;
      }

      this.clearError = function() {
        error.title = '';
        error.style.display = 'none';
      };
      
      this.updateVisibility = function() {
        title.style.color = (self.layer && self.layer.getVisibility() ? '' : '#999');
        visibility.style.color = (self.outOfArea ? '#999' : '');
        visibility.style.cursor = (self.outOfArea ? 'default' : 'pointer');
      };
      
      this.updateLayers = function() {
        var visibleLayers = [];
        for (var i = 0; i < self.mapLayers.length; i++) {
          if (self.mapLayers[i].visible) {
            visibleLayers.push(self.mapLayers[i].name);
          }
        }
        if (visibleLayers && visibleLayers.length == 0 && this.layer) { // Hide map as it has no more layers
          this.layer.setVisibility(false);
        } else if (visibleLayers.length > 0 && !this.layer) { // Add map that received layers
          var params = cloneInto({
            layers: visibleLayers.join(),
            transparent: self.transparent,
            format: map.format
          }, unsafeWindow);
          var options = cloneInto({
            transitionEffect: 'resize',
            attribution: map.attribution,
            isBaseLayer: false
          }, unsafeWindow);
          options.projection = new OL.Projection(map.crs);
          options.tileSize = (map.tile_size ? new OL.Size(map.tile_size, map.tile_size) : new OL.Size(512, 512));
          this.layer = new OL.Layer.WMS(map.title, map.url, params, options);
          this.layer.setOpacity(this.opacity / 100);
          this.layer.setVisibility(!this.hidden && !this.outOfArea);
          this.layer.events.register('tileerror', null, exportFunction(function(obj) {
            if (error.title != '') {
              return;
            }
            error.style.display = 'inline';
            error.title = I18n.t('openmaps.retrieving_error');
            $(error).tooltip();
            loadTileError(obj.tile.url, function(msg) {
              $(error).tooltip('destroy');
              error.title = msg.title + ': ' + msg.description;
              $(error).tooltip();
            });
          }, unsafeWindow));
          this.layer.events.register('tileloadstart', null, exportFunction(function(evt) {
            totalTiles++;
            updateTileLoader();
          }, unsafeWindow));
          this.layer.events.register('tileloaded', null, exportFunction(function(evt) {
            loadedTiles++;
            updateTileLoader();
          }, unsafeWindow));
          Waze.map.addLayer(this.layer);
          var aerialImageryIndex = Waze.map.getLayerIndex(Waze.map.getLayersBy('uniqueName', 'satellite_imagery')[0]);
          Waze.map.setLayerIndex(this.layer, (aerialImageryIndex >= 0 ? aerialImageryIndex : 0) + handles.length + 1);
          this.layer.events.register('visibilitychanged', null, exportFunction(function() {
            self.updateVisibility();
          }, unsafeWindow));
        } else if (layerRedrawNeeded) { // Update layers if there's a change
          this.layer.mergeNewParams(cloneInto({
            layers: visibleLayers.join()
          }, unsafeWindow));
          layerRedrawNeeded = false;
        }
        saveMapState();
      };
      
      remove = createIconButton('', I18n.t('openmaps.remove_layer'));
      remove.addEventListener('click', function(e) {
        Waze.map.removeLayer(self.layer);
        handles.splice(handles.indexOf(self), 1);
        container.parentNode.removeChild(container);
        saveMapState();
      });
      container.appendChild(remove);
      edit = createIconButton('', I18n.t('openmaps.edit_layer'));
      edit.addEventListener('click', function() {
        editContainer.style.display = (editContainer.style.display == 'none' ? 'block' : 'none');
      });
      container.appendChild(edit);
      visibility = createIconButton((self.hidden ? '' : ''), I18n.t('openmaps.hideshow_layer')); // icon-eye-open: , icon-eye-close: 
      visibility.addEventListener('click', function() {
        self.hidden = self.layer.getVisibility();
        self.layer.setVisibility(!self.hidden);
        saveMapState();
      });
      visibility.addEventListener('mouseenter', function() {
        visibility.textContent = (self.hidden ? '' : '');
      });
      visibility.addEventListener('mouseleave', function() {
        visibility.textContent = (self.hidden ? '' : '');
      });
      container.appendChild(visibility);
      if (map.queryable) {
        query = createIconButton('', I18n.t('openmaps.query_layer'));
        query.addEventListener('click', function() {
          this.style.color = 'blue';
          var queryLayers = [];
          self.mapLayers.forEach(function(mapLayer) {
            if (mapLayer.visible && map.layers[mapLayer.name].queryable) {
              queryLayers.push(mapLayer.name);
            }
          });
          queryParams = {
            url: map.url,
            layers: queryLayers.join(),
            callback: function() {
              query.style.color = '';
              document.getElementById('WazeMap').style.cursor = '';
            }
          };
          getFeatureInfoControl.activate();
          document.getElementById('WazeMap').style.cursor = 'help';
        });
        container.appendChild(query);
      }
      error = createIconButton('');
      error.style.color = 'red';
      error.style.display = 'none';
      error.addEventListener('click', function() {
        this.title = '';
        this.style.display = 'none';
      });
      container.appendChild(error);
      title.className = 'title';
      var handle = document.createElement('span');
      handle.style.fontFamily = 'FontAwesome';
      handle.style.letterSpacing = '1px';
      handle.style.color = '#c2c2c2';
      handle.style.cursor = 'move';
      handle.style.fontSize = '11px';
      handle.appendChild(document.createTextNode(' '));
      title.appendChild(handle);
      title.style.cursor = 'default';
      title.style.borderTop = '2px solid transparent';
      title.style.borderWidth = '2px 0 0 0';
      title.appendChild(document.createTextNode(map.title));
      container.appendChild(title);
      description.className = 'additional-info';
      description.style.fontStyle = 'italic';
      description.style.whiteSpace = 'nowrap';
      description.style.textOverflow = 'ellipsis';
      description.style.overflow = 'hidden';
      description.style.cursor = 'pointer';
      description.title = I18n.t('openmaps.expand');
      description.addEventListener('click', function() {
        this.title = (this.style.whiteSpace == 'nowrap' ? I18n.t('openmaps.collapse') : I18n.t('openmaps.expand'));
        this.style.whiteSpace = (this.style.whiteSpace == 'nowrap' ? 'normal' : 'nowrap');
      });
      description.appendChild(document.createTextNode(map.abstract));
      container.appendChild(description);
      var opacityLabel = document.createElement('span');
      opacityLabel.appendChild(document.createTextNode(I18n.t('openmaps.opacity_label') + ':'));
      opacityLabel.style.marginRight = '5px';
      opacityLabel.title = I18n.t('openmaps.opacity_label_tooltip');
      $(opacityLabel).tooltip();
      editContainer.appendChild(opacityLabel);
      var opacitySlider = document.createElement('input');
      opacitySlider.type = 'range';
      opacitySlider.max = 100;
      opacitySlider.min = 5;
      opacitySlider.step = 5;
      opacitySlider.value = this.opacity;
      opacitySlider.style.verticalAlign = 'middle';
      opacitySlider.addEventListener('input', function() {
        self.layer.setOpacity(this.value / 100);
        self.opacity = this.value;
        saveMapState();
      });
      editContainer.appendChild(opacitySlider);
      if (map.format != 'image/jpeg') {
        var transparentContainer = document.createElement('div');
        transparentContainer.className = 'controls-container';
        var checkId = idGenerator.getNext();
        var transparentCheck = document.createElement('input');
        transparentCheck.type = 'checkbox';
        transparentCheck.checked = this.transparent;
        transparentCheck.id = 'transparent' + checkId;
        transparentCheck.addEventListener('click', exportFunction(function() {
          self.layer.mergeNewParams(cloneInto({
            transparent: this.checked
          }, unsafeWindow));
          self.transparent = this.checked;
          saveMapState();
        }, unsafeWindow));
        transparentContainer.appendChild(transparentCheck);
        var transparentLabel = document.createElement('label');
        transparentLabel.title = I18n.t('openmaps.transparent_label_tooltip');
        $(transparentLabel).tooltip();
        transparentLabel.htmlFor = 'transparent' + checkId;
        transparentLabel.appendChild(document.createTextNode(I18n.t('openmaps.transparent_label')));
        transparentContainer.appendChild(transparentLabel);
        editContainer.appendChild(transparentContainer);
      }
      layerContainer.style.overflowY = 'auto';
      layerContainer.style.maxHeight = '15em';
      layerContainer.className = 'result-list';
      self.mapLayers.forEach(function(layerItem) {
        var mapLayer = map.layers[layerItem.name];
        var item = document.createElement('li');
        item.className = 'result';
        var layerHandle = document.createElement('span');
        var layerTitle = document.createElement('span');
        var visibility = document.createElement('span');
        visibility = createIconButton('');
        visibility.addEventListener("click", exportFunction(function() {
          var subjectLayer = self.mapLayers.find(function(mapLayer) { return mapLayer.name == layerItem.name; });
          subjectLayer.visible = !subjectLayer.visible;
          layerTitle.style.color = (subjectLayer.visible ? '' : '#999');
          layerRedrawNeeded = true;
          self.updateLayers.call(self);
        }, unsafeWindow));
        item.appendChild(visibility);
        if (mapLayer.queryable) {
          var layerQuery = document.createElement('span');
          layerQuery = createIconButton('');
          layerQuery.addEventListener('click', function() {
            this.style.color = 'blue';
            queryParams = {
              url: map.url,
              layers: layerItem.name,
              callback: function() {
                layerQuery.style.color = '';
                document.getElementById('WazeMap').style.cursor = '';
              }
            };
            getFeatureInfoControl.activate();
            document.getElementById('WazeMap').style.cursor = 'help';
          });
          item.appendChild(layerQuery);
        }
        layerHandle.style.fontFamily = 'FontAwesome';
        layerHandle.style.letterSpacing = '1px';
        layerHandle.style.color = '#c2c2c2';
        layerHandle.style.cursor = 'move';
        layerHandle.style.fontSize = '11px';
        layerHandle.appendChild(document.createTextNode(' '));
        layerTitle.appendChild(layerHandle);
        layerTitle.className = 'title';
        layerTitle.style.cursor = 'default';
        layerTitle.style.color = (layerItem.visible ? '' : '#999');
        layerTitle.appendChild(document.createTextNode(mapLayer.title));
        layerTitle.addEventListener('click', exportFunction(function(e) {
          visibility.dispatchEvent(new MouseEvent('click'));
        }, unsafeWindow));
        item.appendChild(layerTitle);
        if (mapLayer.abstract) {
          var description = document.createElement('p');
          description.className = 'additional-info';
          description.style.whiteSpace = 'nowrap';
          description.style.textOverflow = 'ellipsis';
          description.style.overflow = 'hidden';
          description.style.cursor = 'pointer';
          description.title = I18n.t('openmaps.expand');
          description.addEventListener('click', function() {
            this.title = (this.style.whiteSpace == 'nowrap' ? I18n.t('openmaps.collapse') : I18n.t('openmaps.expand'));
            this.style.whiteSpace = (this.style.whiteSpace == 'nowrap' ? 'normal' : 'nowrap');
          });
          description.appendChild(document.createTextNode(mapLayer.abstract));
          item.appendChild(description);
        }
        layerContainer.appendChild(item);
      });
      editContainer.appendChild(layerContainer);
      editContainer.style.display = 'none';
      container.appendChild(editContainer);
      container.className = 'result';
      handleList.appendChild(container);
      $(handleList).sortable(); // refresh HTML5Sortable
      $(layerContainer).sortable(cloneInto({
        forcePlaceholderSize: true,
        placeholderClass: 'result',
        handle: '.title'
      }, unsafeWindow)).bind('sortupdate', exportFunction(function(e, ui) {
        if (ui.elementIndex < 0 || ui.elementIndex >= self.mapLayers.length || ui.oldElementIndex < 0 || ui.oldElementIndex >= self.mapLayers.length) { // Sanity check
          log('Received an invalid element index when reordering map layers. Old index: ' + ui.oldElementIndex + ', new index: ' + ui.elementIndex);
          return;
        }
        self.mapLayers.splice(ui.elementIndex, 0, self.mapLayers.splice(ui.oldElementIndex, 1)[0]);
        layerRedrawNeeded = self.mapLayers[ui.elementIndex].visible; // Only redraw if it was a visible layer
        self.updateLayers.call(self);
      }, unsafeWindow));
      
      this.updateLayers();
      this.updateVisibility();
    }
  }

  function log(message) {
    if (typeof message === 'string') {
      console.log('%cWME Open Maps: %c' + message, 'color:black', 'color:#d97e00');
    } else {
      console.log('%cWME Open Maps:', 'color:black', message);
    }
  }

  init();
})();