Greasy Fork is available in English.

WME GAIA + Otros Mapas

WME Ir a Gaia, Google Maps. Redirecciona a la página de cada servicio en la posición actual haciendo clic en la etiqueta de coordenadas de WME.

// ==UserScript==
// @name         WME GAIA + Otros Mapas
// @name:es      WME GAIA - Todos al GAIA
// @namespace    https://greasyfork.org/es/users/1362250-gwm
// @version      1.4
// @description  WME Ir a Gaia, Google Maps. Redirecciona a la página de cada servicio en la posición actual haciendo clic en la etiqueta de coordenadas de WME.
// @description:es WME Go to Gaia, Google Maps te lleva a la posición en el mapa de cada servicio para que no tengas que hacer tantos clicks.
// @author       GWM_
// @match        https://www.waze.com/editor*
// @match        https://www.waze.com/*/editor*
// @match        https://beta.waze.com/*
// @match        *://www.google.com/maps*
// @match        *://maps.google.com*
// @match        https://gaia.inegi.org.mx/mdm6
// @exclude      https://www.waze.com/user/editor*
// @exclude      https://www.waze.com/*/user/editor*
// @require      https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @require      https://greasyfork.org/scripts/383120-proj4-wazedev/code/proj4-Wazedev.js
// @grant           GM_xmlhttpRequest
// @noframes
// @namespace https://greasyfork.org/es/users/1362250-gwm
// @contributionURL https://github.com/WazeDev/Thank-The-Authors
// @license      GPLv3
// ==/UserScript==

/* global $ */
/* global OpenLayers */
/* global WazeWrap */
/* global I18n */
/* global W */
/* global proj4 */
/* ecmaVersion 2017 */
/* eslint curly: ["warn", "multi-or-nest"] */

(function() {
    'use strict';
    //var jqUI_CssSrc = GM_getResourceText("jqUI_CSS");
    //GM_addStyle(jqUI_CssSrc);
    var settings = {};
    var wazerIcon = "";
    var gmapsIcon = "";
    var GMDMIcon = ""

    function initInterface(){
        var $section = $("<div>");
        $section.html([
            '<fieldset style="border: 1px solid silver; padding: 8px; border-radius: 4px;">',
            "<p>Los Siguientes mapas son <span style='color:red; font-weight:bold;'>Sólo Referencias</span> y <b>No deben Copiarse Datos</b> ya que se viola la política de fuentes externas de Waze.</p>",
            `<div><input type="checkbox" id="chkGMaps" class="OOMchk"><label for="chkGMaps"><img src="${gmapsIcon}" height="18" width="18">Google Maps</label></div>`,
            `<div><input type="checkbox" id="chkGMDM" class="OOMchk"><label for="chkGMDM"><img src="${GMDMIcon}" height="18" width="18">Gaia - Mexico</label></div>`,
            '</br><div></fieldset>',

            '<fieldset style="border: 1px solid silver; padding: 8px; border-radius: 4px;">',
            '<legend style="margin-bottom:0px; border-bottom-style:none; width:auto;"><h4>Idioma del mapa (cuando corresponda)</h4></legend>',
            '<input type="radio" name="radOOMLanguage" id="radOOMNoLang">No establezca un idioma</br>',
            '<input type="radio" name="radOOMLanguage" id="radOOMWMELang">Utilice el lenguaje WME</br>',
            '<input type="radio" name="radOOMLanguage" id="radOOMCustomLang">Seleccionar Lenguaje <input type="text" name="txtOOMLanguage" id="txtOOMLanguage" style="border: 1px solid #000000;" size="4"/>',
            '</fieldset>',
            '</div>',
            '<div><fieldset style="border: 1px solid silver; padding: 8px; border-radius: 4px;">',
            '<legend style="margin-bottom: 0px; border-bottom-style:none; width: auto;"><h4>Superposición de marcadores de Google Maps</h4></legend>',
            'Link del mapa: <input type="text" name="txtOOMMyMapLink" id="txtOOMMyMapLink"/>',
            '<button id="OOMLoadMyMap">Cargar Mapa</button>',
            '</fieldset></div>',
            '</div>'
        ].join(' '));

        WazeWrap.Interface.Tab('MAPSMEX', $section.html(), init, 'MAPSMEX');
    }

    function getolControlAttributionDivRightValue(){
        return parseInt($('.wz-map-ol-control-attribution').css("right").slice(0,-2));;
    }

    function init(){
        loadSettings();
        setChecked('chkGMaps', settings.GMaps);
        setChecked('chkGMDM', settings.GMDM);

        if(settings.LangSetting == 0)
            setChecked("radOOMNoLang", true);
        else if(settings.LangSetting == 1)
            setChecked("radOOMWMELang", true);
        else
            setChecked("radOOMCustomLang", true);

        $('#txtOOMLanguage')[0].value = settings.CustLang;

        let annoyingDivRight = getolControlAttributionDivRightValue();
        $('.wz-map-ol-control-attribution').css("right", `${annoyingDivRight+100}px`);
        annoyingDivRight = getolControlAttributionDivRightValue();
        let checkedBoxes = $('.OOMchk:Checked');
        let totalButtonsWidth = 0;
        for(let i=0; i<checkedBoxes.length;i++){
            totalButtonsWidth += parseInt($(`label[for='${$(checkedBoxes[i]).attr('id')}'] img`).css('width').slice(0,-2));
        }
        $('.wz-map-ol-control-attribution').css("right", `${annoyingDivRight+totalButtonsWidth}px`);

        LoadMapButtons();
        $('.OOMchk').change(function() {
            var settingName = $(this)[0].id.substr(3);
            settings[settingName] = this.checked;
            saveSettings();
            LoadMapButtons();

            let btnWidth = parseInt($(`label[for='${$(this).attr('id')}'] img`).css('width').slice(0,-2));
            if(this.checked){ //add button width
                let annoyingDivRight = getolControlAttributionDivRightValue();
                $('.wz-map-ol-control-attribution').css("right", `${annoyingDivRight+btnWidth}px`);
            }
            else{ //subtract button width
                let annoyingDivRight = getolControlAttributionDivRightValue();
                $('.wz-map-ol-control-attribution').css("right", `${annoyingDivRight-btnWidth}px`);
            }
        });
        $("[id^='rad']").change(function() {
            if(isChecked("radOOMNoLang"))
                settings.LangSetting = 0;
            else if(isChecked("radOOMWMELang"))
                settings.LangSetting = 1;
            else
                settings.LangSetting = 2;
            saveSettings();
        });
        $('#txtOOMLanguage').focusout(function(){
            settings.CustLang = $('#txtOOMLanguage').val();
            saveSettings();
        });
        $('#OOMLoadMyMap').click(loadMyMap);
        injectOLMyMapKML();
        WazeWrap.Interface.ShowScriptUpdate("WME GAIA + Otros Mapas", GM_info.script.version, updateMessage, "https://greasyfork.org/es/scripts/508007-wme-gaia-otros-mapas");
    }

    async function getKML(url){
        try{
            return await wrapGMXMLHTTP(url);
        }
        catch(err){
            let patt = new RegExp(/^<\?xml(?:.|\n)*<\/kml>/);
            let res = patt.test(err.responseText);
            if(res)
                return err.responseText;
            else
                console.log("Error retrieving the MyMap data\n\n" + err);
        }
    }

    async function wrapGMXMLHTTP(url){
        return new Promise((resolve, reject) => {
            GM_xmlhttpRequest({
                url,
                method: 'GET',
                onload(res) {
                    if (res.status < 400) {
                        resolve(res.responseText);
                    } else {
                        console.log("error");
                        console.log(res.status);
                        // handle errors here
                    }
                },
                onerror(res) {
                    // handle errors here
                    console.log("Error:");
                    console.log(res.text);
                }
            });
        });
    }

    async function loadMyMap(){
        let url = $('#txtOOMMyMapLink')[0].value;
        debugger;
        if(!url.length > 0)
            return;
        let patt = new RegExp(/^(?:http(s)?:\/\/)?www.google.com\/maps+[\w\-\._~:\/?#[\]%@!\$&\'\(\)\*\+,;=.]+$/);
        let res = patt.test(url);
        if(!res){ //not a google mymap url
            WazeWrap.Alerts.error(GM_info.script.name, "Esta no es una URL válida de Google MyMap");
            return;
        }

        let mid = url.match(/mid=(.*?)(&|$)/)[1];
        let mapKML = await getKML(`https://www.google.com/maps/d/kml?mid=${mid}&forcekml=1`);
        let parser = new OpenLayers.Format.MyMapKML();
        parser.extractStyles = true;
        parser.internalProjection = W.Config.map.projection.local;
        parser.externalProjection = new OpenLayers.Projection("EPSG:4326");

        if(W.map.getLayersByName("Google MyMap").length > 0)
            W.map.removeLayer(W.map.getLayersByName("Google MyMap")[0]);
        var OOMMyMapLayer = new OpenLayers.Layer.Vector("Google MyMap", { rendererOptions: { zIndexing: true }, uniqueName: "wme_oommymap", layerGroup: 'wme_oommymap'});
        OOMMyMapLayer.setZIndex(-9999);

        var features;
        if(mapKML.documentElement)
            features = parser.read(new XMLSerializer().serializeToString(mapKML.documentElement));
        else
            features = parser.read(mapKML);

        // check which attribute can be used for labels
        /*let maxlabels = 5000;
        var labelname = /^description|description$/;
        if (features.length <= maxlabels) {
            for (var attr in features[0].attributes) {
                if (labelname.test(attr.toLowerCase()) === true) {
                    if (typeof features[0].attributes[attr] == 'string') {
                        //layerStyle.label = '${'+attr+'}';
                        break;
                    }
                }
            }
        }*/

        OOMMyMapLayer.addFeatures(features);
        W.map.addLayer(OOMMyMapLayer);
        WazeWrap.Interface.AddLayerCheckbox("display", "Google MyMap", true, function(visible){OOMMyMapLayer.setVisibility(visible);});
    }

    function GetLanguage(){
        if(isChecked("radOOMNoLang"))
            return "";
        else if(isChecked("radOOMWMELang"))
            return I18n.currentLocale().replace("es-419", "es");
        else //Custom Language
            return $('#txtOOMLanguage').val();
    }

    function get4326CenterPoint(){
        let projI = new OpenLayers.Projection("EPSG:900913");
        let projE = new OpenLayers.Projection("EPSG:4326");
        let center_lonlat = (new OpenLayers.LonLat(W.map.getCenter().lon, W.map.getCenter().lat)).transform(projI,projE);
        let lat = Math.round(center_lonlat.lat * 1000000) / 1000000;
        let lon = Math.round(center_lonlat.lon * 1000000) / 1000000;
        return new OpenLayers.LonLat(lon, lat);
    }

    var insertPath = '.WazeControlPermalink';
    function LoadMapButtons(){
        $('#OOMGMaps').remove();
        //********************* Google Maps *********************
        if(settings.GMaps){
            let $section = $("<div>", {style:"padding:8px 16px"});
            $section.html([
                '<span id="OOMGMaps">',
                `<img src="${gmapsIcon}" alt="Google Maps" width="18" height="18" id="OOMGMapsImg" title="Abrir en Google Maps" style="cursor:pointer; float: left; display:inline-block; margin: 2px 5px 0 3px;">`,
                '</span>'
            ].join(' '));

            $(insertPath).prepend($section.html());

            $('#OOMGMapsImg').click(function(){
                let latlon = get4326CenterPoint();
                let lang = GetLanguage();

                window.open('https://www.google.com/maps/@' + latlon.lat + ',' + latlon.lon + ',' + ( W.map.getZoom()) + 'z' + (lang != "" ? "?hl=" + lang : ""), 'Google Maps');
            });
        }

        //********************* GAIA - INEGI *********************

        $('#OOMGMDM').remove();
        if(settings.GMDM){
            let $sectionGMDM = $("<div>");
            $sectionGMDM.html([
                '<span id="OOMGMDM">',
                `<img src="${GMDMIcon}" alt="Gaia Mexico" width="18" height="18" id="OOMGMDMImg" title="Abrir en GAIA Mexico" style="cursor:pointer; float: left; display:inline-block; margin: 2px 5px 0 3px;">`,
                '</span>'
            ].join(' '));

            $(insertPath).prepend($sectionGMDM.html());
            $('#OOMGMDMImg').click(function(){
                let latlon = get4326CenterPoint();

                window.open(`http://gaia.inegi.org.mx/mdm6/?v=${btoa("lat:"+latlon.lat+",lon:"+latlon.lon+",z:"+(W.map.getZoom()+8))}`);
            });
        }
 
    }

        function loadSettings() {
            var loadedSettings = $.parseJSON(localStorage.getItem("OOM_Settings"));
            var defaultSettings = {
                GMaps: false,
                Bing: false,
                OSM: false,
                LangSetting: 1,
                CustLang: "",
                Here: false,
                GMDM: false,
            };
            settings = loadedSettings ? loadedSettings : defaultSettings;
            for (var prop in defaultSettings) {
                if (!settings.hasOwnProperty(prop))
                    settings[prop] = defaultSettings[prop];
            }
        }

        function saveSettings() {
            if (localStorage) {
                var localsettings = {
                    GMaps: settings.GMaps,
                    Bing: settings.Bing,
                    OSM: settings.OSM,
                    LangSetting: settings.LangSetting,
                    CustLang: settings.CustLang,
                    Here: settings.Here,
                    GMDM: settings.GMDM,
                };

                localStorage.setItem("OOM_Settings", JSON.stringify(localsettings));
            }
        }

        function isChecked(checkboxId) {
            return $('#' + checkboxId).is(':checked');
        }

        function setChecked(checkboxId, checked) {
            $('#' + checkboxId).prop('checked', checked);
        }

        function bootstrapGeneral(initdelegate, tries = 1){
            if(document.readyState !== 'complete' )
                setTimeout(function() {bootstrapGeneral(initdelegate, tries++);}, 200);
            else
                initdelegate();
        }

        function bootstrap(tries = 1) {
            if(location.href.indexOf("google.com/maps") > -1)
                bootstrapGeneral(initGoogleMaps, 1);
            /*else if(location.href.indexOf("http://www.511nj.org/trafficmap") > -1){
            bootstrapGeneral(initNJ511, 1);
        }*/
            else{
                if (W &&
                    W.map &&
                    W.model &&
                    $ && WazeWrap.Ready) {
                    initInterface();
                } else if (tries < 1000) {
                    setTimeout(function () {bootstrap(tries++);}, 200);
                }
            }
        }

        function RosreestrToWaze(){
            let lon, lat, zoom;
            let curURL = location.href.match(/x=(\d*.\d*)&y=(\d*.\d*)&z=(\d+)/);
            lon = curURL[1];
            lat = curURL[2];
            zoom = parseInt(curURL[3]);
 
            let source = new proj4.Proj('EPSG:900913');
 
            var point = new proj4.toPoint([parseFloat(lon), parseFloat(lat)]);
            point = proj4.transform(source, proj4.WGS84, point);
            return `https://www.waze.com/en-US/editor/?lon=${point.x}&lat=${point.y}&zoom=${(Math.max(0,Math.min(10,(zoom - 12))))}`;
        }
 
        function initRosreestr(){
            var observer = new MutationObserver(function(mutations) {
                mutations.forEach(function(mutation) {
                    if (mutation.type === "attributes" && mutation.target == document.getElementsByClassName("btn btn-default btn-tool-lg js-showList")[0]) insertWMELinkRosreestr();
                });
            });
 
            observer.observe(document.getElementById("sidebar-region"), { childList: true, subtree: true, attributes:true});
 
            insertWMELinkRosreestr();
        }
 
        /* //Waze ink Español 419 punto de compartir
            return `https://www.waze.com/es-419/editor?env=row&lat=${point.y}&lon=${point.x}&zoomLevel=${(Math.max(0,Math.min(10,(zoom - 12))))}`;
            */

        function initGoogleMaps(){
            let $OOMWazeButton = document.createElement("div");
            $OOMWazeButton.innerHTML = `<div id="OOMWazeButtonDiv" style="height:36px; width:36px; position: fixed; right:30px; top:75px; cursor: pointer; background-image: url(${wazerIcon}); background-size: 36px 36px; background-repeat: no-repeat;" title="Abrir en MAPSMEX WME"></div>`;
            let parent = document.getElementById("content-container");
            parent.appendChild($OOMWazeButton);

            document.getElementById("OOMWazeButtonDiv").addEventListener("click", function(){
                window.open(GMToWaze());
            });

            document.getElementById('OOMWazeButtonDiv').addEventListener("mouseenter",function(e) {
                document.addEventListener('keydown', copyPLHotkeyEvent);
                document.getElementsByClassName('widget-scene-canvas')[0].addEventListener('keydown', copyPLHotkeyEvent);
            });

            document.getElementById('OOMWazeButtonDiv').addEventListener('mouseleave', function() {
                document.removeEventListener('keydown', copyPLHotkeyEvent);
                document.getElementsByClassName('widget-scene-canvas')[0].removeEventListener('keydown', copyPLHotkeyEvent);
            });
        }

        var copyToClipboard = function(str) {
            var temp = document.createElement("input");
            document.body.append(temp);
            temp.value = str;
            temp.select();
            document.execCommand('copy');
            document.body.removeChild(temp);
        };

        var copyPLHotkeyEvent = function(e) {
            if ((e.metaKey || e.ctrlKey) && (e.which === 67))
                copyToClipboard(GMToWaze());
        };

        function GMToWaze(){
            let lon, lat, zoom;
            let curURL = location.href.split('@').pop().split(',');
            lon = curURL[1];
            lat = curURL[0];
            zoom = parseInt(curURL[2]);
            if(zoom > 21)
                zoom = 17;
            return `https://www.waze.com/en-US/editor/?lon=${lon}&lat=${lat}&zoom=${(Math.max(0,Math.min(10,(zoom - 12))))}`;
        }

        function injectOLMyMapKML(){
            if(!OpenLayers.Format.MyMapKML){
                OpenLayers.Format.MyMapKML = OpenLayers.Class(OpenLayers.Format.XML,{namespaces:{kml:"http://www.opengis.net/kml/2.2",gx:"http://www.google.com/kml/ext/2.2"},kmlns:"http://earth.google.com/kml/2.0",placemarksDesc:"No description available",foldersName:"OpenLayers export",foldersDesc:"Exported on "+new Date,extractAttributes:!0,kvpAttributes:!1,extractStyles:!1,extractTracks:!1,trackAttributes:null,internalns:null,features:null,styles:null,styleBaseUrl:"",fetched:null,maxDepth:0,iconColorMap:{"#880e4f":"","#a52714":"","#e65100":"","#f9a825":"","#ffd600":"","#817717":"","#558b2f":"","#097138":"","#006064":"","#01579b":"","#1a237e":"","#673ab7":"","#4e342e":"","#c2185b":"","#ff5252":"","#f57c00":"","#fbc02d":"","#ffea00":"","#afb42b":"","#7cb342":"","#0f9d58":"","#0097a7":"","#0288d1":"","#3949ab":"","#9c27b0":"","#795548":"","#bdbdbd":"","#757575":"","#424242":"","#000000":""},initialize:function(A){this.regExes={trimSpace:/^\s*|\s*$/g,removeSpace:/\s*/g,splitSpace:/\s+/,trimComma:/\s*,\s*/g,kmlColor:/(\w{2})(\w{2})(\w{2})(\w{2})/,kmlIconPalette:/root:\/\/icons\/palette-(\d+)(\.\w+)/,straightBracket:/\$\[(.*?)\]/g},this.externalProjection=new OpenLayers.Projection("EPSG:4326"),OpenLayers.Format.XML.prototype.initialize.apply(this,[A])},read:function(A){this.features=[],this.styles={},this.fetched={};var e={depth:0,styleBaseUrl:this.styleBaseUrl};return this.parseData(A,e)},parseData:function(A,e){"string"==typeof A&&(A=OpenLayers.Format.XML.prototype.read.apply(this,[A]));for(var t=["Link","NetworkLink","Style","StyleMap","Placemark"],a=0,i=t.length;a<i;++a){var s=t[a],n=this.getElementsByTagNameNS(A,"*",s);if(0!=n.length)switch(s.toLowerCase()){case"link":case"networklink":this.parseLinks(n,e);break;case"style":this.extractStyles&&this.parseStyles(n,e);break;case"stylemap":this.extractStyles&&this.parseStyleMaps(n,e);break;case"placemark":this.parseFeatures(n,e)}}return this.features},parseLinks:function(A,e){if(e.depth>=this.maxDepth)return!1;var t=OpenLayers.Util.extend({},e);t.depth++;for(var a=0,i=A.length;a<i;a++){var s=this.parseProperty(A[a],"*","href");if(s&&!this.fetched[s]){this.fetched[s]=!0;var n=this.fetchLink(s);n&&this.parseData(n,t)}}},fetchLink:function(A){var e=OpenLayers.Request.GET({url:A,async:!1});if(e)return e.responseText},parseStyles:function(A,e){for(var t=0,a=A.length;t<a;t++){var i=this.parseStyle(A[t]);if(i){var s=(e.styleBaseUrl||"")+"#"+i.id;this.styles[s]=i}}},parseKmlColor:function(A){var e=null;if(A){var t=A.match(this.regExes.kmlColor);t&&(e={color:"#"+t[4]+t[3]+t[2],opacity:parseInt(t[1],16)/255,r:parseInt(t[4],16),g:parseInt(t[3],16),b:parseInt(t[2],16)})}return e},parseStyle:function(A){for(var e,t,a={},i=["LineStyle","PolyStyle","IconStyle","BalloonStyle","LabelStyle"],s=0,n=i.length;s<n;++s)if(e=i[s],t=this.getElementsByTagNameNS(A,"*",e)[0]){var r=this.parseProperty(t,"*","color"),h=this.parseKmlColor(r);switch(e.toLowerCase()){case"linestyle":h&&(a.strokeColor=h.color,a.strokeOpacity=h.opacity),(g=this.parseProperty(t,"*","width"))&&(a.strokeWidth=g);break;case"polystyle":h&&(a.fillOpacity=h.opacity,a.fillColor=h.color),"0"==this.parseProperty(t,"*","fill")&&(a.fillColor="none"),"0"==this.parseProperty(t,"*","outline")&&(a.strokeWidth="0");break;case"iconstyle":var o=parseFloat(this.parseProperty(t,"*","scale")||1),g=32*o,l=32*o,E=this.getElementsByTagNameNS(t,"*","Icon")[0];if(E){var B=this.parseProperty(E,"*","href");if(B){var d=this.parseProperty(E,"*","w"),c=this.parseProperty(E,"*","h");!OpenLayers.String.startsWith(B,"http://maps.google.com/mapfiles/kml")||d||c||(d=64,c=64,o/=2),d=d||c,c=c||d,d&&(g=parseInt(d)*o),c&&(l=parseInt(c)*o);var Q=B.match(this.regExes.kmlIconPalette);if(Q){var C=Q[1],f=Q[2],N=this.parseProperty(E,"*","x"),R=this.parseProperty(E,"*","y");B="http://maps.google.com/mapfiles/kml/pal"+C+"/icon"+(8*(R?7-R/32:7)+(N?N/32:0))+f}a.graphicOpacity=1,a.externalGraphic=this.iconColorMap[h.color],a.graphicYOffset=-32}}a.graphicWidth=g,a.graphicHeight=l;break;case"balloonstyle":var v=OpenLayers.Util.getXmlNodeValue(t);v&&(a.balloonStyle=v.replace(this.regExes.straightBracket,"${$1}"));break;case"labelstyle":r=this.parseProperty(t,"*","color");(h=this.parseKmlColor(r))&&(a.fontColor=h.color,a.fontOpacity=h.opacity)}}!a.strokeColor&&a.fillColor&&(a.strokeColor=a.fillColor);var u=A.getAttribute("id");return u&&a&&(a.id=u),a},parseStyleMaps:function(A,e){for(var t=0,a=A.length;t<a;t++)for(var i=A[t],s=this.getElementsByTagNameNS(i,"*","Pair"),n=i.getAttribute("id"),r=0,h=s.length;r<h;r++){var o=s[r],g=this.parseProperty(o,"*","key"),l=this.parseProperty(o,"*","styleUrl");l&&"normal"==g&&(this.styles[(e.styleBaseUrl||"")+"#"+n]=this.styles[(e.styleBaseUrl||"")+l])}},parseFeatures:function(A,e){for(var t=[],a=0,i=A.length;a<i;a++){var s=A[a],n=this.parseFeature.apply(this,[s]);if(!n)throw"Bad Placemark: "+a;if(this.extractStyles&&n.attributes&&n.attributes.styleUrl&&(n.style=this.getStyle(n.attributes.styleUrl,e)),this.extractStyles){var r=this.getElementsByTagNameNS(s,"*","Style")[0];if(r){var h=this.parseStyle(r);h&&(n.style=OpenLayers.Util.extend(n.style,h))}}if(this.extractTracks){var o=this.getElementsByTagNameNS(s,this.namespaces.gx,"Track");if(o&&o.length>0){var g=o[0],l={features:[],feature:n};this.readNode(g,l),l.features.length>0&&t.push.apply(t,l.features)}}else t.push(n)}this.features=this.features.concat(t)},readers:{kml:{when:function(A,e){e.whens.push(OpenLayers.Date.parse(this.getChildValue(A)))},_trackPointAttribute:function(A,e){var t=A.nodeName.split(":").pop();e.attributes[t].push(this.getChildValue(A))}},gx:{Track:function(A,e){var t={whens:[],points:[],angles:[]};if(this.trackAttributes){t.attributes={};for(var a=0,i=this.trackAttributes.length;a<i;++a)l=this.trackAttributes[a],t.attributes[l]=[],l in this.readers.kml||(this.readers.kml[l]=this.readers.kml._trackPointAttribute)}if(this.readChildNodes(A,t),t.whens.length!==t.points.length)throw new Error("gx:Track with unequal number of when ("+t.whens.length+") and gx:coord ("+t.points.length+") elements.");var s,n,r,h=t.angles.length>0;if(h&&t.whens.length!==t.angles.length)throw new Error("gx:Track with unequal number of when ("+t.whens.length+") and gx:angles ("+t.angles.length+") elements.");for(a=0,i=t.whens.length;a<i;++a){if((s=e.feature.clone()).fid=e.feature.fid||e.feature.id,n=t.points[a],s.geometry=n,"z"in n&&(s.attributes.altitude=n.z),this.internalProjection&&this.externalProjection&&s.geometry.transform(this.externalProjection,this.internalProjection),this.trackAttributes)for(var o=0,g=this.trackAttributes.length;o<g;++o){var l=this.trackAttributes[o];s.attributes[l]=t.attributes[l][a]}s.attributes.when=t.whens[a],s.attributes.trackId=e.feature.id,h&&(r=t.angles[a],s.attributes.heading=parseFloat(r[0]),s.attributes.tilt=parseFloat(r[1]),s.attributes.roll=parseFloat(r[2])),e.features.push(s)}},coord:function(A,e){var t=this.getChildValue(A).replace(this.regExes.trimSpace,"").split(/\s+/),a=new OpenLayers.Geometry.Point(t[0],t[1]);t.length>2&&(a.z=parseFloat(t[2])),e.points.push(a)},angles:function(A,e){var t=this.getChildValue(A).replace(this.regExes.trimSpace,"").split(/\s+/);e.angles.push(t)}}},parseFeature:function(A){for(var e,t,a,i,s=["MultiGeometry","Polygon","LineString","Point"],n=0,r=s.length;n<r;++n)if(e=s[n],this.internalns=A.namespaceURI?A.namespaceURI:this.kmlns,(t=this.getElementsByTagNameNS(A,this.internalns,e)).length>0){var h;if(!(h=this.parseGeometry[e.toLowerCase()]))throw new TypeError("Unsupported geometry type: "+e);a=h.apply(this,[t[0]]),this.internalProjection&&this.externalProjection&&a.transform(this.externalProjection,this.internalProjection);break}this.extractAttributes&&(i=this.parseAttributes(A));var o=new OpenLayers.Feature.Vector(a,i),g=A.getAttribute("id")||A.getAttribute("name");return null!=g&&(o.fid=g),o},getStyle:function(A,e){var t=OpenLayers.Util.removeTail(A),a=OpenLayers.Util.extend({},e);if(a.depth++,a.styleBaseUrl=t,!this.styles[A]&&!OpenLayers.String.startsWith(A,"#")&&a.depth<=this.maxDepth&&!this.fetched[t]){var i=this.fetchLink(t);i&&this.parseData(i,a)}return OpenLayers.Util.extend({},this.styles[A])},parseGeometry:{point:function(A){var e=this.getElementsByTagNameNS(A,this.internalns,"coordinates"),t=[];if(e.length>0){var a=e[0].firstChild.nodeValue;t=(a=a.replace(this.regExes.removeSpace,"")).split(",")}if(!(t.length>1))throw"Bad coordinate string: "+a;return 2==t.length&&(t[2]=null),new OpenLayers.Geometry.Point(t[0],t[1],t[2])},linestring:function(A,e){var t=this.getElementsByTagNameNS(A,this.internalns,"coordinates"),a=null;if(t.length>0){for(var i,s=this.getChildValue(t[0]),n=(s=(s=s.replace(this.regExes.trimSpace,"")).replace(this.regExes.trimComma,",")).split(this.regExes.splitSpace),r=n.length,h=new Array(r),o=0;o<r;++o){if(!((i=n[o].split(",")).length>1))throw"Bad LineString point coordinates: "+n[o];2==i.length&&(i[2]=null),h[o]=new OpenLayers.Geometry.Point(i[0],i[1],i[2])}if(!r)throw"Bad LineString coordinates: "+s;a=e?new OpenLayers.Geometry.LinearRing(h):new OpenLayers.Geometry.LineString(h)}return a},polygon:function(A){var e=this.getElementsByTagNameNS(A,this.internalns,"LinearRing"),t=e.length,a=new Array(t);if(t>0)for(var i,s=0,n=e.length;s<n;++s){if(!(i=this.parseGeometry.linestring.apply(this,[e[s],!0])))throw"Bad LinearRing geometry: "+s;a[s]=i}return new OpenLayers.Geometry.Polygon(a)},multigeometry:function(A){for(var e,t=[],a=A.childNodes,i=0,s=a.length;i<s;++i)if(1==(e=a[i]).nodeType){var n,r=e.prefix?e.nodeName.split(":")[1]:e.nodeName;(n=this.parseGeometry[r.toLowerCase()])&&t.push(n.apply(this,[e]))}return new OpenLayers.Geometry.Collection(t)}},parseAttributes:function(A){var e,t,a={},i=A.getElementsByTagName("ExtendedData");i.length&&(a=this.parseExtendedData(i[0]));for(var s=A.childNodes,n=0,r=s.length;n<r;++n)if(1==(e=s[n]).nodeType&&(t=e.childNodes).length>=1&&t.length<=3){var h;switch(t.length){case 1:h=t[0];break;case 2:var o=t[0],g=t[1];h=3==o.nodeType||4==o.nodeType?o:g;break;case 3:default:h=t[1]}if(3==h.nodeType||4==h.nodeType){var l=e.prefix?e.nodeName.split(":")[1]:e.nodeName,E=OpenLayers.Util.getXmlNodeValue(h);E&&(E=E.replace(this.regExes.trimSpace,""),a[l]=E)}}return a},parseExtendedData:function(A){var e,t,a,i,s={},n=A.getElementsByTagName("Data");for(e=0,t=n.length;e<t;e++){i=(a=n[e]).getAttribute("name");var r={},h=a.getElementsByTagName("value");if(h.length&&(r.value=this.getChildValue(h[0])),this.kvpAttributes)s[i]=r.value;else{var o=a.getElementsByTagName("displayName");o.length&&(r.displayName=this.getChildValue(o[0])),s[i]=r}}var g=A.getElementsByTagName("SimpleData");for(e=0,t=g.length;e<t;e++){r={};i=(a=g[e]).getAttribute("name"),r.value=this.getChildValue(a),this.kvpAttributes?s[i]=r.value:(r.displayName=i,s[i]=r)}return s},parseProperty:function(A,e,t){var a,i=this.getElementsByTagNameNS(A,e,t);try{a=OpenLayers.Util.getXmlNodeValue(i[0])}catch(A){a=null}return a},write:function(A){OpenLayers.Util.isArray(A)||(A=[A]);for(var e=this.createElementNS(this.kmlns,"kml"),t=this.createFolderXML(),a=0,i=A.length;a<i;++a)t.appendChild(this.createPlacemarkXML(A[a]));return e.appendChild(t),OpenLayers.Format.XML.prototype.write.apply(this,[e])},createFolderXML:function(){var A=this.createElementNS(this.kmlns,"Folder");if(this.foldersName){var e=this.createElementNS(this.kmlns,"name"),t=this.createTextNode(this.foldersName);e.appendChild(t),A.appendChild(e)}if(this.foldersDesc){var a=this.createElementNS(this.kmlns,"description"),i=this.createTextNode(this.foldersDesc);a.appendChild(i),A.appendChild(a)}return A},createPlacemarkXML:function(A){var e=this.createElementNS(this.kmlns,"name"),t=A.style&&A.style.label?A.style.label:A.id,a=A.attributes.name||t;e.appendChild(this.createTextNode(a));var i=this.createElementNS(this.kmlns,"description"),s=A.attributes.description||this.placemarksDesc;i.appendChild(this.createTextNode(s));var n=this.createElementNS(this.kmlns,"Placemark");null!=A.fid&&n.setAttribute("id",A.fid),n.appendChild(e),n.appendChild(i);var r=this.buildGeometryNode(A.geometry);if(n.appendChild(r),A.attributes){var h=this.buildExtendedData(A.attributes);h&&n.appendChild(h)}return n},buildGeometryNode:function(A){var e=A.CLASS_NAME,t=e.substring(e.lastIndexOf(".")+1),a=this.buildGeometry[t.toLowerCase()],i=null;return a&&(i=a.apply(this,[A])),i},buildGeometry:{point:function(A){var e=this.createElementNS(this.kmlns,"Point");return e.appendChild(this.buildCoordinatesNode(A)),e},multipoint:function(A){return this.buildGeometry.collection.apply(this,[A])},linestring:function(A){var e=this.createElementNS(this.kmlns,"LineString");return e.appendChild(this.buildCoordinatesNode(A)),e},multilinestring:function(A){return this.buildGeometry.collection.apply(this,[A])},linearring:function(A){var e=this.createElementNS(this.kmlns,"LinearRing");return e.appendChild(this.buildCoordinatesNode(A)),e},polygon:function(A){for(var e,t,a,i=this.createElementNS(this.kmlns,"Polygon"),s=A.components,n=0,r=s.length;n<r;++n)a=0==n?"outerBoundaryIs":"innerBoundaryIs",e=this.createElementNS(this.kmlns,a),t=this.buildGeometry.linearring.apply(this,[s[n]]),e.appendChild(t),i.appendChild(e);return i},multipolygon:function(A){return this.buildGeometry.collection.apply(this,[A])},collection:function(A){for(var e,t=this.createElementNS(this.kmlns,"MultiGeometry"),a=0,i=A.components.length;a<i;++a)(e=this.buildGeometryNode.apply(this,[A.components[a]]))&&t.appendChild(e);return t}},buildCoordinatesNode:function(A){var e,t=this.createElementNS(this.kmlns,"coordinates"),a=A.components;if(a){for(var i,s=a.length,n=new Array(s),r=0;r<s;++r)i=a[r],n[r]=this.buildCoordinates(i);e=n.join(" ")}else e=this.buildCoordinates(A);var h=this.createTextNode(e);return t.appendChild(h),t},buildCoordinates:function(A){return this.internalProjection&&this.externalProjection&&(A=A.clone()).transform(this.internalProjection,this.externalProjection),A.x+","+A.y},buildExtendedData:function(A){var e=this.createElementNS(this.kmlns,"ExtendedData");for(var t in A)if(A[t]&&"name"!=t&&"description"!=t&&"styleUrl"!=t){var a=this.createElementNS(this.kmlns,"Data");a.setAttribute("name",t);var i=this.createElementNS(this.kmlns,"value");if("object"==typeof A[t]){if(A[t].value&&i.appendChild(this.createTextNode(A[t].value)),A[t].displayName){var s=this.createElementNS(this.kmlns,"displayName");s.appendChild(this.getXMLDoc().createCDATASection(A[t].displayName)),a.appendChild(s)}}else i.appendChild(this.createTextNode(A[t]));a.appendChild(i),e.appendChild(a)}return this.isSimpleContent(e)?null:e},changeImageColor:function(A,e,t,a){function i(A,e,t){A/=255,e/=255,t/=255;let a,i,s=Math.max(A,e,t),n=Math.min(A,e,t),r=(s+n)/2;if(s==n)a=i=0;else{let h=s-n;switch(i=r>.5?h/(2-s-n):h/(s+n),s){case A:a=(e-t)/h+(e<t?6:0);break;case e:a=(t-A)/h+2;break;case t:a=(A-e)/h+4}a/=6}return{h:a,s:i,l:r}}let s=document.createElement("canvas").getContext("2d");s.drawImage(A,0,0);let n=s.getImageData(0,0,32,32),r=n.data;for(var h=0;h<r.length;h+=4){let A=r[h+0],s=r[h+1],n=r[h+2];r[h+3]<10||360*i(A,s,n).h<20&&(r[h+0]=e,r[h+1]=t,r[h+2]=a)}let o=document.createElement("canvas");return $(o).attr("width",32),$(o).attr("height",32),o.getContext("2d").putImageData(n,0,0),o.toDataURL()},CLASS_NAME:"OpenLayers.Format.KML"});
            }
        }


        bootstrap();
    })();