This repository was archived by the owner on Jun 10, 2023. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathmaplibre-gl-vector-text-protocol.min.js
More file actions
2 lines (2 loc) · 157 KB
/
maplibre-gl-vector-text-protocol.min.js
File metadata and controls
2 lines (2 loc) · 157 KB
1
2
!function(g,C){"object"==typeof exports&&"undefined"!=typeof module?C(exports):"function"==typeof define&&define.amd?define(["exports"],C):C((g="undefined"!=typeof globalThis?globalThis:g||self).VectorTextProtocol={})}(this,(function(g){"use strict";function C(g){return g&&g.__esModule&&Object.prototype.hasOwnProperty.call(g,"default")?g.default:g}function I(g){if(g.__esModule)return g;var C=g.default;if("function"==typeof C){var I=function g(){if(this instanceof g){var I=[null];return I.push.apply(I,arguments),new(Function.bind.apply(C,I))}return C.apply(this,arguments)};I.prototype=C.prototype}else I={};return Object.defineProperty(I,"__esModule",{value:!0}),Object.keys(g).forEach((function(C){var A=Object.getOwnPropertyDescriptor(g,C);Object.defineProperty(I,C,A.get?A:{enumerable:!0,get:function(){return g[C]}})})),I}function A(g){return new Function("d","return {"+g.map((function(g,C){return JSON.stringify(g)+": d["+C+"]"})).join(",")+"}")}function l(g){var C=new RegExp('["'+g+"\n]"),I=g.charCodeAt(0);function l(g,C){var A,l,b={},c={},o=[],Z=g.length,d=0,G=0;function s(){if(d>=Z)return c;if(l)return l=!1,b;var C,A=d;if(34===g.charCodeAt(A)){for(var o=A;o++<Z;)if(34===g.charCodeAt(o)){if(34!==g.charCodeAt(o+1))break;++o}return d=o+2,13===(C=g.charCodeAt(o+1))?(l=!0,10===g.charCodeAt(o+2)&&++d):10===C&&(l=!0),g.slice(A+1,o).replace(/""/g,'"')}for(;d<Z;){var G=1;if(10===(C=g.charCodeAt(d++)))l=!0;else if(13===C)l=!0,10===g.charCodeAt(d)&&(++d,++G);else if(C!==I)continue;return g.slice(A,d-G)}return g.slice(A)}for(;(A=s())!==c;){for(var e=[];A!==b&&A!==c;)e.push(A),A=s();C&&null==(e=C(e,G++))||o.push(e)}return o}function b(C){return C.map(c).join(g)}function c(g){return null==g?"":C.test(g+="")?'"'+g.replace(/\"/g,'""')+'"':g}return{parse:function(g,C){var I,b,c=l(g,(function(g,l){if(I)return I(g,l-1);b=g,I=C?function(g,C){var I=A(g);return function(A,l){return C(I(A),l,g)}}(g,C):A(g)}));return c.columns=b,c},parseRows:l,format:function(C,I){return null==I&&(I=function(g){var C=Object.create(null),I=[];return g.forEach((function(g){for(var A in g)A in C||I.push(C[A]=A)})),I}(C)),[I.map(c).join(g)].concat(C.map((function(C){return I.map((function(g){return c(C[g])})).join(g)}))).join("\n")},formatRows:function(g){return g.map(b).join("\n")}}}var b=l(","),c=b.parse,o=b.parseRows,Z=b.format,d=b.formatRows,G=l("\t"),s=G.parse,e=G.parseRows,n=G.format,t=G.formatRows,m=I(Object.freeze({__proto__:null,dsvFormat:l,csvParse:c,csvParseRows:o,csvFormat:Z,csvFormatRows:d,tsvParse:s,tsvParseRows:e,tsvFormat:n,tsvFormatRows:t})),B={exports:{}};function i(g,C){var I=W(g,C);return I.whole+"° "+(I.minutes?I.minutes+"' ":"")+(I.seconds?I.seconds+'" ':"")+I.dir}function W(g,C){var I=({lat:["N","S"],lon:["E","W"]}[C]||"")[g>=0?0:1],A=Math.abs(g),l=Math.floor(A),b=60*(A-l),c=Math.floor(b);return{whole:l,minutes:c,seconds:Math.floor(60*(b-c)),dir:I}}function J(g,C){if(C||(C="NSEW"),"string"!=typeof g)return null;var I=(g=g.toUpperCase()).match(/^[\s\,]*([NSEW])?\s*([\-|\—|\―]?[0-9.]+)[°º˚]?\s*(?:([0-9.]+)['’′‘]\s*)?(?:([0-9.]+)(?:''|"|”|″)\s*)?([NSEW])?/);if(!I)return null;var A,l=I[0];if(I[1]&&I[5]?(A=I[1],l=l.slice(0,-1)):A=I[1]||I[5],A&&-1===C.indexOf(A))return null;var b=I[2]?parseFloat(I[2]):0,c=I[3]?parseFloat(I[3])/60:0,o=I[4]?parseFloat(I[4])/3600:0,Z=b<0?-1:1;return"S"!==A&&"W"!==A||(Z*=-1),{val:(Math.abs(b)+c+o)*Z,dim:A,matched:l,remain:g.slice(l.length)}}B.exports=function(g,C){var I=J(g,C);return null===I?null:I.val},B.exports.pair=function(g,C){var I=J(g=g.trim(),C);if(!I)return null;var A=J(g=I.remain.trim(),C);if(!A||A.remain)return null;return I.dim?function(g,C,I){if("N"===I||"S"===I)return[g,C];if("W"===I||"E"===I)return[C,g]}(I.val,A.val,I.dim):[I.val,A.val]},B.exports.format=i,B.exports.formatPair=function(g){return i(g.lat,"lat")+" "+i(g.lon,"lon")},B.exports.coordToDMS=W;var a=B.exports,S=m,K=a,k=/(Lat)(itude)?/gi,u=/(L)(on|ng)(gitude)?/i;function h(g,C){var I,A,l;for(var b in g)(A=b.match(C))&&(!I||A[0].length/b.length>l)&&(l=A[0].length/b.length,I=b);return I}function y(g){return h(g,k)}function V(g){return h(g,u)}function X(g){return"object"==typeof g?Object.keys(g).length:0}function p(g){var C=[];return[",",";","\t","|"].forEach((function(I){var A=S.dsvFormat(I).parse(g);if(A.length>=1){for(var l=X(A[0]),b=0;b<A.length;b++)if(X(A[b])!==l)return;C.push({delimiter:I,arity:Object.keys(A[0]).length})}})),C.length?C.sort((function(g,C){return C.arity-g.arity}))[0].delimiter:null}var H={isLon:function(g){return!!g.match(u)},isLat:function(g){return!!g.match(k)},guessLatHeader:y,guessLonHeader:V,csv:S.csvParse,tsv:S.tsvParse,dsv:S,auto:function(g){var C=p(g);return C?function(g){return delete g.columns,g}(S.dsvFormat(C).parse(g)):null},csv2geojson:function(g,C,I){I||(I=C,C={}),C.delimiter=C.delimiter||",";var A=C.latfield||"",l=C.lonfield||"",b=C.crs||"",c=[],o={type:"FeatureCollection",features:c};if(""!==b&&(o.crs={type:"name",properties:{name:b}}),"auto"!==C.delimiter||"string"!=typeof g||(C.delimiter=p(g),C.delimiter)){var Z=C.numericFields?C.numericFields.split(","):null,d="string"==typeof g?S.dsvFormat(C.delimiter).parse(g,(function(g){if(Z)for(var C in g)Z.includes(C)&&(g[C]=+g[C]);return g})):g;if(d.length){var G,s=[];if(A||(A=y(d[0])),l||(l=V(d[0])),!A||!l){for(G=0;G<d.length;G++)c.push({type:"Feature",properties:d[G],geometry:null});I(s.length?s:null,o)}else{for(G=0;G<d.length;G++)if(void 0!==d[G][l]&&void 0!==d[G][A]){var e,n,t,m=d[G][l],B=d[G][A];(t=K(m,"EW"))&&(m=t),(t=K(B,"NS"))&&(B=t),e=parseFloat(m),n=parseFloat(B),isNaN(e)||isNaN(n)?s.push({message:"A row contained an invalid value for latitude or longitude",row:d[G],index:G}):(C.includeLatLon||(delete d[G][l],delete d[G][A]),c.push({type:"Feature",properties:d[G],geometry:{type:"Point",coordinates:[parseFloat(e),parseFloat(n)]}}))}I(s.length?s:null,o)}}else I(null,o)}else I({type:"Error",message:"Could not autodetect delimiter"})},toLine:function(g){for(var C=g.features,I={type:"Feature",geometry:{type:"LineString",coordinates:[]}},A=0;A<C.length;A++)I.geometry.coordinates.push(C[A].geometry.coordinates);return I.properties=C.reduce((function(g,C){for(var I in C.properties)g[I]||(g[I]=[]),g[I].push(C.properties[I]);return g}),{}),{type:"FeatureCollection",features:[I]}},toPolygon:function(g){for(var C=g.features,I={type:"Feature",geometry:{type:"Polygon",coordinates:[[]]}},A=0;A<C.length;A++)I.geometry.coordinates[0].push(C[A].geometry.coordinates);return I.properties=C.reduce((function(g,C){for(var I in C.properties)g[I]||(g[I]=[]),g[I].push(C.properties[I]);return g}),{}),{type:"FeatureCollection",features:[I]}}};function Y(g){return g}function R(g,C){var I=C.id,A=C.bbox,l=null==C.properties?{}:C.properties,b=function(g,C){var I=function(g){if(null==g)return Y;var C,I,A=g.scale[0],l=g.scale[1],b=g.translate[0],c=g.translate[1];return function(g,o){o||(C=I=0);var Z=2,d=g.length,G=new Array(d);for(G[0]=(C+=g[0])*A+b,G[1]=(I+=g[1])*l+c;Z<d;)G[Z]=g[Z],++Z;return G}}(g.transform),A=g.arcs;function l(g,C){C.length&&C.pop();for(var l=A[g<0?~g:g],b=0,c=l.length;b<c;++b)C.push(I(l[b],b));g<0&&function(g,C){for(var I,A=g.length,l=A-C;l<--A;)I=g[l],g[l++]=g[A],g[A]=I}(C,c)}function b(g){return I(g)}function c(g){for(var C=[],I=0,A=g.length;I<A;++I)l(g[I],C);return C.length<2&&C.push(C[0]),C}function o(g){for(var C=c(g);C.length<4;)C.push(C[0]);return C}function Z(g){return g.map(o)}function d(g){var C,I=g.type;switch(I){case"GeometryCollection":return{type:I,geometries:g.geometries.map(d)};case"Point":C=b(g.coordinates);break;case"MultiPoint":C=g.coordinates.map(b);break;case"LineString":C=c(g.arcs);break;case"MultiLineString":C=g.arcs.map(c);break;case"Polygon":C=Z(g.arcs);break;case"MultiPolygon":C=g.arcs.map(Z);break;default:return null}return{type:I,coordinates:C}}return d(C)}(g,C);return null==I&&null==A?{type:"Feature",properties:l,geometry:b}:null==A?{type:"Feature",id:I,properties:l,geometry:b}:{type:"Feature",id:I,bbox:A,properties:l,geometry:b}}function r(g,C){return Array.from(g.getElementsByTagName(C))}function w(g){return"#"===g[0]?g:`#${g}`}function v(g){return g?.normalize(),g&&g.textContent||""}function F(g,C,I){const A=g.getElementsByTagName(C),l=A.length?A[0]:null;return l&&I&&I(l),l}function N(g,C,I){const A={};if(!g)return A;const l=g.getElementsByTagName(C),b=l.length?l[0]:null;return b&&I?I(b,A):A}function z(g,C,I){const A=v(F(g,C));return A&&I&&I(A)||{}}function L(g,C,I){const A=parseFloat(v(F(g,C)));if(!isNaN(A))return A&&I&&I(A)||{}}function f(g,C,I){const A=parseFloat(v(F(g,C)));if(!isNaN(A))return A&&I&&I(A),A}function x(g,C){const I={};for(const A of C)z(g,A,(g=>{I[A]=g}));return I}function Q(g){return 1===g?.nodeType}function T(g){return N(g,"line",(g=>Object.assign({},z(g,"color",(g=>({stroke:`#${g}`}))),L(g,"opacity",(g=>({"stroke-opacity":g}))),L(g,"width",(g=>({"stroke-width":96*g/25.4}))))))}function U(g){let C=[];if(null===g)return C;for(const I of Array.from(g.childNodes)){if(!Q(I))continue;const g=M(I.nodeName);if("gpxtpx:TrackPointExtension"===g)C=C.concat(U(I));else{const A=v(I);C.push([g,O(A)])}}return C}function M(g){return["heart","gpxtpx:hr","hr"].includes(g)?"heart":g}function O(g){const C=parseFloat(g);return isNaN(C)?g:C}function j(g){const C=[parseFloat(g.getAttribute("lon")||""),parseFloat(g.getAttribute("lat")||"")];if(isNaN(C[0])||isNaN(C[1]))return null;f(g,"ele",(g=>{C.push(g)}));const I=F(g,"time");return{coordinates:C,time:I?v(I):null,extendedValues:U(F(g,"extensions"))}}function D(g){const C=x(g,["name","cmt","desc","type","time","keywords"]),I=Array.from(g.getElementsByTagNameNS("http://www.garmin.com/xmlschemas/GpxExtensions/v3","*"));for(const A of I)A.parentNode?.parentNode===g&&(C[A.tagName.replace(":","_")]=v(A));const A=r(g,"link");return A.length&&(C.links=A.map((g=>Object.assign({href:g.getAttribute("href")},x(g,["text","type"]))))),C}function P(g,C){const I=r(g,C),A=[],l=[],b={};for(let g=0;g<I.length;g++){const C=j(I[g]);if(C){A.push(C.coordinates),C.time&&l.push(C.time);for(const[A,l]of C.extendedValues){const C="heart"===A?A:A.replace("gpxtpx:","")+"s";b[C]||(b[C]=Array(I.length).fill(null)),b[C][g]=l}}}if(!(A.length<2))return{line:A,times:l,extendedValues:b}}function E(g){const C=P(g,"rtept");if(C)return{type:"Feature",properties:Object.assign({_gpxType:"rte"},D(g),T(F(g,"extensions"))),geometry:{type:"LineString",coordinates:C.line}}}function q(g){const C=r(g,"trkseg"),I=[],A=[],l=[];for(const g of C){const C=P(g,"trkpt");C&&(l.push(C),C.times&&C.times.length&&A.push(C.times))}if(0===l.length)return null;const b=l.length>1,c=Object.assign({_gpxType:"trk"},D(g),T(F(g,"extensions")),A.length?{coordinateProperties:{times:b?A:A[0]}}:{});for(const g of l){I.push(g.line),c.coordinateProperties||(c.coordinateProperties={});const C=c.coordinateProperties,A=Object.entries(g.extendedValues);for(let g=0;g<A.length;g++){const[I,c]=A[g];b?(C[I]||(C[I]=l.map((g=>new Array(g.line.length).fill(null)))),C[I][g]=c):C[I]=c}}return{type:"Feature",properties:c,geometry:b?{type:"MultiLineString",coordinates:I}:{type:"LineString",coordinates:I[0]}}}function _(g){const C=Object.assign(D(g),x(g,["sym"])),I=j(g);return I?{type:"Feature",properties:C,geometry:{type:"Point",coordinates:I.coordinates}}:null}function*$(g){for(const C of r(g,"trk")){const g=q(C);g&&(yield g)}for(const C of r(g,"rte")){const g=E(C);g&&(yield g)}for(const C of r(g,"wpt")){const g=_(C);g&&(yield g)}}const gg="http://www.garmin.com/xmlschemas/ActivityExtension/v2",Cg=[["heartRate","heartRates"],["Cadence","cadences"],["Speed","speeds"],["Watts","watts"]],Ig=[["TotalTimeSeconds","totalTimeSeconds"],["DistanceMeters","distanceMeters"],["MaximumSpeed","maxSpeed"],["AverageHeartRateBpm","avgHeartRate"],["MaximumHeartRateBpm","maxHeartRate"],["AvgSpeed","avgSpeed"],["AvgWatts","avgWatts"],["MaxWatts","maxWatts"]];function Ag(g,C){const I=[];for(const[A,l]of C){let C=F(g,A);if(!C){const I=g.getElementsByTagNameNS(gg,A);I.length&&(C=I[0])}const b=parseFloat(v(C));isNaN(b)||I.push([l,b])}return I}function lg(g){const C=[f(g,"LongitudeDegrees"),f(g,"LatitudeDegrees")];if(void 0===C[0]||isNaN(C[0])||void 0===C[1]||isNaN(C[1]))return null;const I=F(g,"HeartRateBpm"),A=v(F(g,"Time"));return F(g,"AltitudeMeters",(g=>{const I=parseFloat(v(g));isNaN(I)||C.push(I)})),{coordinates:C,time:A||null,heartRate:I?parseFloat(v(I)):null,extensions:Ag(g,Cg)}}function bg(g){const C=r(g,"Trackpoint"),I=[],A=[],l=[];if(C.length<2)return null;const b={},c={extendedProperties:b};for(let g=0;g<C.length;g++){const c=lg(C[g]);if(null===c)continue;I.push(c.coordinates);const{time:o,heartRate:Z,extensions:d}=c;o&&A.push(o),Z&&l.push(Z);for(const[I,A]of d)b[I]||(b[I]=Array(C.length).fill(null)),b[I][g]=A}return I.length<2?null:Object.assign(c,{line:I,times:A,heartRates:l})}function cg(g){const C=r(g,"Track"),I=[],A=[],l=[],b=[];let c;const o=Object.assign(Object.fromEntries(Ag(g,Ig)),N(g,"Name",(g=>({name:v(g)}))));for(const g of C)c=bg(g),c&&(I.push(c.line),c.times.length&&A.push(c.times),c.heartRates.length&&l.push(c.heartRates),b.push(c.extendedProperties));for(let g=0;g<b.length;g++){const A=b[g];for(const l in A)1===C.length?c&&(o[l]=c.extendedProperties[l]):(o[l]||(o[l]=I.map((g=>Array(g.length).fill(null)))),o[l][g]=A[l])}return 0===I.length?null:((A.length||l.length)&&(o.coordinateProperties=Object.assign(A.length?{times:1===I.length?A[0]:A}:{},l.length?{heart:1===I.length?l[0]:l}:{})),{type:"Feature",properties:o,geometry:1===I.length?{type:"LineString",coordinates:I[0]}:{type:"MultiLineString",coordinates:I}})}function*og(g){for(const C of r(g,"Lap")){const g=cg(C);g&&(yield g)}for(const C of r(g,"Courses")){const g=cg(C);g&&(yield g)}}function Zg(g,C){const I={},A="stroke"==C||"fill"===C?C:C+"-color";return"#"===g[0]&&(g=g.substring(1)),6===g.length||3===g.length?I[A]="#"+g:8===g.length&&(I[C+"-opacity"]=parseInt(g.substring(0,2),16)/255,I[A]="#"+g.substring(6,8)+g.substring(4,6)+g.substring(2,4)),I}function dg(g,C,I){const A={};return f(g,C,(g=>{A[I]=g})),A}function Gg(g,C){return N(g,"color",(g=>Zg(v(g),C)))}function sg(g){return N(g,"Icon",((g,C)=>(z(g,"href",(g=>{C.icon=g})),C)))}function eg(g){return Object.assign({},function(g){return N(g,"PolyStyle",((g,C)=>Object.assign(C,N(g,"color",(g=>Zg(v(g),"fill"))),z(g,"fill",(g=>{if("0"===g)return{"fill-opacity":0}})),z(g,"outline",(g=>{if("0"===g)return{"stroke-opacity":0}})))))}(g),function(g){return N(g,"LineStyle",(g=>Object.assign(Gg(g,"stroke"),dg(g,"width","stroke-width"))))}(g),function(g){return N(g,"LabelStyle",(g=>Object.assign(Gg(g,"label"),dg(g,"scale","label-scale"))))}(g),function(g){return N(g,"IconStyle",(g=>Object.assign(Gg(g,"icon"),dg(g,"scale","icon-scale"),dg(g,"heading","icon-heading"),N(g,"hotSpot",(g=>{const C=parseFloat(g.getAttribute("x")||""),I=parseFloat(g.getAttribute("y")||""),A=g.getAttribute("xunits")||"",l=g.getAttribute("yunits")||"";return isNaN(C)||isNaN(I)?{}:{"icon-offset":[C,I],"icon-offset-units":[A,l]}})),sg(g))))}(g))}const ng=g=>Number(g),tg={string:g=>g,int:ng,uint:ng,short:ng,ushort:ng,float:ng,double:ng,bool:g=>Boolean(g)};function mg(g,C){return N(g,"ExtendedData",((g,I)=>{for(const C of r(g,"Data"))I[C.getAttribute("name")||""]=v(F(C,"value"));for(const A of r(g,"SimpleData")){const g=A.getAttribute("name")||"",l=C[g]||tg.string;I[g]=l(v(A))}return I}))}function Bg(g){const C=F(g,"description");for(const g of Array.from(C?.childNodes||[]))if(4===g.nodeType)return{description:{"@type":"html",value:v(g)}};return{}}function ig(g){return N(g,"TimeSpan",(g=>({timespan:{begin:v(F(g,"begin")),end:v(F(g,"end"))}})))}function Wg(g){return N(g,"TimeStamp",(g=>({timestamp:v(F(g,"when"))})))}function Jg(g,C){return z(g,"styleUrl",(g=>(g=w(g),C[g]?Object.assign({styleUrl:g},C[g]):{styleUrl:g})))}const ag=/\s*/g,Sg=/^\s*|\s*$/g,Kg=/\s+/;function kg(g){return g.replace(ag,"").split(",").map(parseFloat).filter((g=>!isNaN(g))).slice(0,3)}function ug(g){return g.replace(Sg,"").split(Kg).map(kg).filter((g=>g.length>=2))}function hg(g){let C=r(g,"coord");0===C.length&&(C=function(g,C,I){return Array.from(g.getElementsByTagNameNS(I,C))}(g,"coord","*"));const I=C.map((g=>v(g).split(" ").map(parseFloat)));return 0===I.length?null:{geometry:I.length>2?{type:"LineString",coordinates:I}:{type:"Point",coordinates:I[0]},times:r(g,"when").map((g=>v(g)))}}function yg(g){if(0===g.length)return g;const C=g[0],I=g[g.length-1];let A=!0;for(let g=0;g<Math.max(C.length,I.length);g++)if(C[g]!==I[g]){A=!1;break}return A?g:g.concat([g[0]])}function Vg(g){return v(F(g,"coordinates"))}function Xg(g){let C=[],I=[];for(let A=0;A<g.childNodes.length;A++){const l=g.childNodes.item(A);if(Q(l))switch(l.tagName){case"MultiGeometry":case"MultiTrack":case"gx:MultiTrack":{const g=Xg(l);C=C.concat(g.geometries),I=I.concat(g.coordTimes);break}case"Point":{const g=kg(Vg(l));g.length>=2&&C.push({type:"Point",coordinates:g});break}case"LinearRing":case"LineString":{const g=ug(Vg(l));g.length>=2&&C.push({type:"LineString",coordinates:g});break}case"Polygon":{const g=[];for(const C of r(l,"LinearRing")){const I=yg(ug(Vg(C)));I.length>=4&&g.push(I)}g.length&&C.push({type:"Polygon",coordinates:g});break}case"Track":case"gx:Track":{const g=hg(l);if(!g)break;const{times:A,geometry:b}=g;C.push(b),A.length&&I.push(A);break}}}return{geometries:C,coordTimes:I}}function pg(g){return 0===g.length?null:1===g.length?g[0]:{type:"GeometryCollection",geometries:g}}function Hg(g,C,I){const{coordTimes:A,geometries:l}=Xg(g),b={type:"Feature",geometry:pg(l),properties:Object.assign(x(g,["name","address","visibility","open","phoneNumber","description"]),Bg(g),Jg(g,C),eg(g),mg(g,I),ig(g),Wg(g),A.length?{coordinateProperties:{times:1===A.length?A[0]:A}}:{})};void 0!==b.properties?.visibility&&(b.properties.visibility="0"!==b.properties.visibility);const c=g.getAttribute("id");return null!==c&&""!==c&&(b.id=c),b}function Yg(g){if(F(g,"gx:LatLonQuad")){return{type:"Polygon",coordinates:[yg(ug(Vg(g)))]}}return function(g){const C=F(g,"LatLonBox");if(C){const g=f(C,"north"),I=f(C,"west"),A=f(C,"east"),l=f(C,"south"),b=f(C,"rotation");if("number"==typeof g&&"number"==typeof l&&"number"==typeof I&&"number"==typeof A){let C=[[[I,g],[A,g],[A,l],[I,l],[I,g]]];return"number"==typeof b&&(C=function(g,C,I){const A=[(g[0]+g[2])/2,(g[1]+g[3])/2];return[C[0].map((g=>{const C=g[1]-A[1],l=g[0]-A[0],b=Math.sqrt(Math.pow(C,2)+Math.pow(l,2)),c=Math.atan2(C,l)-I*Rg;return[A[0]+Math.cos(c)*b,A[1]+Math.sin(c)*b]}))]}([I,l,A,g],C,b)),{type:"Polygon",coordinates:C}}}return null}(g)}const Rg=Math.PI/180;function rg(g,C,I){const A={type:"Feature",geometry:Yg(g),properties:Object.assign({"@geometry-type":"groundoverlay"},x(g,["name","address","visibility","open","phoneNumber","description"]),Bg(g),Jg(g,C),eg(g),sg(g),mg(g,I),ig(g),Wg(g))};void 0!==A.properties?.visibility&&(A.properties.visibility="0"!==A.properties.visibility);const l=g.getAttribute("id");return null!==l&&""!==l&&(A.id=l),A}function wg(g){let C=g.getAttribute("id");const I=g.parentNode;return!C&&Q(I)&&"CascadingStyle"===I.localName&&(C=I.getAttribute("kml:id")||I.getAttribute("id")),w(C||"")}function vg(g){const C={};for(const I of r(g,"Style"))C[wg(I)]=eg(I);for(const I of r(g,"StyleMap")){const g=w(I.getAttribute("id")||"");z(I,"styleUrl",(I=>{I=w(I),C[I]&&(C[g]=C[I])}))}return C}function Fg(g){const C={};for(const I of r(g,"SimpleField"))C[I.getAttribute("name")||""]=tg[I.getAttribute("type")||""]||tg.string;return C}const Ng=["name","visibility","open","address","description","phoneNumber","visibility"];function*zg(g){const C=vg(g),I=Fg(g);for(const A of r(g,"Placemark")){const g=Hg(A,C,I);g&&(yield g)}for(const A of r(g,"GroundOverlay")){const g=rg(A,C,I);g&&(yield g)}}var Lg=Object.freeze({__proto__:null,gpx:function(g){return{type:"FeatureCollection",features:Array.from($(g))}},gpxGen:$,kml:function(g){return{type:"FeatureCollection",features:Array.from(zg(g))}},kmlGen:zg,kmlWithFolders:function(g){const C=vg(g),I=Fg(g),A={type:"root",children:[]};return function g(A,l){if(Q(A))switch(A.tagName){case"GroundOverlay":{const g=rg(A,C,I);g&&l.children.push(g);break}case"Placemark":{const g=Hg(A,C,I);g&&l.children.push(g);break}case"Folder":{const g=function(g){const C={};for(const I of Array.from(g.childNodes))Q(I)&&Ng.includes(I.tagName)&&(C[I.tagName]=v(I));return{type:"folder",meta:C,children:[]}}(A);l.children.push(g),l=g;break}}if(A.childNodes)for(let C=0;C<A.childNodes.length;C++)g(A.childNodes[C],l)}(g,A),A},tcx:function(g){return{type:"FeatureCollection",features:Array.from(og(g))}},tcxGen:og}),fg=(()=>{const g=g=>g[0],C=g=>g[g.length-1],I=g=>g.join(","),A=(g,C,I)=>{let A=g[C];A?A.push(I):g[C]=[I]},l=(g,C,I)=>{let A=g[C],l=null;A&&(l=A.indexOf(I))>=0&&A.splice(l,1)},b=(g,C)=>{let I=g[C];return I&&I.length>0?I[0]:null},c=A=>A.length>3&&I(g(A))===I(C(A)),o=(g,C,I)=>{C=C||0,I=I||1;let A=g.reduce(((I,A,l)=>g[I][C]>A[C]?I:l),0),l=A<=0?g.length-2:A-1,b=A>=g.length-1?1:A+1,c=g[l][C],o=g[A][C],Z=g[b][C],d=g[l][I],G=g[A][I];return(o-c)*(g[b][I]-d)-(Z-c)*(G-d)<0?"clockwise":"counterclockwise"},Z=g=>g instanceof Array?g.map(Z):parseFloat(g);class d extends Map{constructor(){super(),this.binders=[]}add(g,C){this.has(g)||this.set(g,C)}addBinder(g){this.binders.push(g)}bindAll(){this.binders.forEach((g=>g.bind()))}}return{purgeProps:(g,C)=>{if(g){let I=Object.assign({},g);if(C)for(let g of C)delete I[g];return I}return{}},mergeProps:(g,C)=>(g=g||{},C=C||{},Object.assign(g,C)),first:g,last:C,coordsToKey:I,addToMap:A,removeFromMap:l,getFirstFromMap:b,isRing:c,ringDirection:o,ptInsidePolygon:(g,C,I,A)=>{I=I||0,A=A||1;let l=!1;for(let b=0,c=C.length-1;b<C.length;c=b++)(C[b][I]<=g[I]&&g[I]<C[c][I]||C[c][I]<=g[I]&&g[I]<C[b][I])&&g[A]<(C[c][A]-C[b][A])*(g[I]-C[b][I])/(C[c][I]-C[b][I])+C[b][A]&&(l=!l);return l},strToFloat:Z,RefElements:d,LateBinder:class{constructor(g,C,I,A){this.container=g,this.valueFunc=C,this.ctx=I,this.args=A}bind(){let g=this.valueFunc.apply(this.ctx,this.args);if(this.container instanceof Array){let C=this.container.indexOf(this);if(C>=0){let I=[C,1];g&&I.push(g),[].splice.apply(this.container,I)}}else if("object"==typeof this.container){let C=Object.keys(this.container).find((g=>this.container[g]===this));C&&(g?this.container[C]=g:delete this.container[C])}}},WayCollection:class extends Array{constructor(){super(),this.firstMap={},this.lastMap={}}addWay(l){(l=l.toCoordsArray()).length>0&&(this.push(l),A(this.firstMap,I(g(l)),l),A(this.lastMap,I(C(l)),l))}toStrings(){let A=[],c=null;for(;c=this.shift();){l(this.firstMap,I(g(c)),c),l(this.lastMap,I(C(c)),c);let o=c,d=null;do{let A=I(C(o)),c=!1;d=b(this.firstMap,A),d||(d=b(this.lastMap,A),c=!0),d&&(this.splice(this.indexOf(d),1),l(this.firstMap,I(g(d)),d),l(this.lastMap,I(C(d)),d),c&&(d.length>o.length&&([o,d]=[d,o]),d.reverse()),o=o.concat(d.slice(1)))}while(d);A.push(Z(o))}return A}toRings(g){let C=this.toStrings(),I=[],A=null;for(;A=C.shift();)c(A)&&(o(A)!==g&&A.reverse(),I.push(A));return I}}}})(),xg={building:{},highway:{whitelist:["services","rest_area","escape","elevator"]},natural:{blacklist:["coastline","cliff","ridge","arete","tree_row"]},landuse:{},waterway:{whitelist:["riverbank","dock","boatyard","dam"]},amenity:{},leisure:{},barrier:{whitelist:["city_wall","ditch","hedge","retaining_wall","wall","spikes"]},railway:{whitelist:["station","turntable","roundhouse","platform"]},area:{},boundary:{},man_made:{blacklist:["cutline","embankment","pipeline"]},power:{whitelist:["plant","substation","generator","transformer"]},place:{},shop:{},aeroway:{blacklist:["taxiway"]},tourism:{},historic:{},public_transport:{},office:{},"building:part":{},military:{},ruins:{},"area:highway":{},craft:{},golf:{},indoor:{}},Qg=(()=>{const{first:g,last:C,coordsToKey:I,addToMap:A,removeFromMap:l,getFirstFromMap:b,isRing:c,ringDirection:o,ptInsidePolygon:Z,strToFloat:d,LateBinder:G,WayCollection:s}=fg,e=xg;class n{constructor(g,C,I){this.type=g,this.id=C,this.refElems=I,this.tags={},this.props={id:this.getCompositeId()},this.refCount=0,this.hasTag=!1,I&&I.add(this.getCompositeId(),this)}addTags(g){this.tags=Object.assign(this.tags,g),this.hasTag=!!g}addTag(g,C){this.tags[g]=C,this.hasTag=!!g}addProp(g,C){this.props[g]=C}addProps(g){this.props=Object.assign(this.props,g)}getCompositeId(){return`${this.type}/${this.id}`}getProps(){return Object.assign(this.props,this.tags)}toFeatureArray(){return[]}}class t extends n{constructor(g,C){super("node",g,C),this.latLng=null}setLatLng(g){this.latLng=g}toFeatureArray(){return this.latLng?[{type:"Feature",id:this.getCompositeId(),properties:this.getProps(),geometry:{type:"Point",coordinates:d([this.latLng.lon,this.latLng.lat])}}]:[]}getLatLng(){return this.latLng}}class m extends n{constructor(g,C){super("way",g,C),this.latLngArray=[],this.isPolygon=!1}addLatLng(g){this.latLngArray.push(g)}setLatLngArray(g){this.latLngArray=g}addNodeRef(g){let C=new G(this.latLngArray,(function(g){let C=this.refElems.get(`node/${g}`);if(C)return C.refCount++,C.getLatLng()}),this,[g]);this.latLngArray.push(C),this.refElems.addBinder(C)}analyzeTag(g,C){let I=e[g];I&&(this.isPolygon=!0,I.whitelist?this.isPolygon=I.whitelist.indexOf(C)>=0:I.blacklist&&(this.isPolygon=!(I.blacklist.indexOf(C)>=0)))}addTags(g){super.addTags(g);for(let[C,I]of Object.entries(g))this.analyzeTag(C,I)}addTag(g,C){super.addTag(g,C),this.analyzeTag(g,C)}toCoordsArray(){return this.latLngArray.map((g=>[g.lon,g.lat]))}toFeatureArray(){let g=this.toCoordsArray();if(g.length>1){g=d(g);let C={type:"Feature",id:this.getCompositeId(),properties:this.getProps(),geometry:{type:"LineString",coordinates:g}};return this.isPolygon&&c(g)?("counterclockwise"!==o(g)&&g.reverse(),C.geometry={type:"Polygon",coordinates:[g]},[C]):[C]}return[]}}return{Node:t,Way:m,Relation:class extends n{constructor(g,C){super("relation",g,C),this.relations=[],this.nodes=[],this.bounds=null}setBounds(g){this.bounds=g}addMember(g){switch(g.type){case"relation":let C=new G(this.relations,(function(g){let C=this.refElems.get(`relation/${g}`);if(C)return C.refCount++,C}),this,[g.ref]);this.relations.push(C),this.refElems.addBinder(C);break;case"way":g.role||(g.role="");let I=this[g.role];if(I||(I=this[g.role]=[]),g.geometry){let C=new m(g.ref,this.refElems);C.setLatLngArray(g.geometry),C.refCount++,I.push(C)}else if(g.nodes){let C=new m(g.ref,this.refElems);for(let I of g.nodes)C.addNodeRef(I);C.refCount++,I.push(C)}else{let C=new G(I,(function(g){let C=this.refElems.get(`way/${g}`);if(C)return C.refCount++,C}),this,[g.ref]);I.push(C),this.refElems.addBinder(C)}break;case"node":let A=null;if(g.lat&&g.lon){A=new t(g.ref,this.refElems),A.setLatLng({lon:g.lon,lat:g.lat}),g.tags&&A.addTags(g.tags);for(let[C,I]of Object.entries(g))["id","type","lat","lon"].indexOf(C)<0&&A.addProp(C,I);A.refCount++,this.nodes.push(A)}else{let C=new G(this.nodes,(function(g){let C=this.refElems.get(`node/${g}`);if(C)return C.refCount++,C}),this,[g.ref]);this.nodes.push(C),this.refElems.addBinder(C)}}}toFeatureArray(){let C=[],I=[],A=[];const l=["outer","inner",""];for(let g of this.relations)if(g)for(let C of l){let I=g[C];if(I){let g=this[C];g?[].splice.apply(g,[g.length,0].concat(I)):this[C]=I}}for(let g of l){let C=this[g];if(C){this[g]=new s;for(let I of C)this[g].addWay(I)}}let b=null,c={type:"Feature",id:this.getCompositeId(),bbox:this.bounds,properties:this.getProps()};this.bounds||delete c.bbox,this.outer?(b=((C,I)=>{let A=C?C.toRings("counterclockwise"):[],l=I?I.toRings("clockwise"):[];if(A.length>0){let C=[],I=null;for(I of A)C.push([I]);for(;I=l.shift();)for(let l in A)if(Z(g(I),A[l])){C[l].push(I);break}return 1===C.length?{type:"Polygon",coordinates:C[0]}:{type:"MultiPolygon",coordinates:C}}return null})(this.outer,this.inner),b&&(c.geometry=b,C.push(c))):this[""]&&(b=(g=>{let C=g?g.toStrings():[];return C.length>0?1===C.length?{type:"LineString",coordinates:C[0]}:{type:"MultiLineString",coordinates:C}:null})(this[""]),b&&(c.geometry=b,I.push(c)));for(let g of this.nodes)A=A.concat(g.toFeatureArray());return C.concat(I).concat(A)}}}})(),Tg=(()=>{function g(g){return null!=g.match(/^(.+?)\[(.+?)\]>$/g)}function C(g){let C=/^(.+?)\[(.+?)\]>$/g.exec(g);return C?{evt:C[1]+">",exp:C[2]}:{evt:g}}return class{constructor(g){g&&(this.queryParent=!!g.queryParent,this.progressive=g.progressive,this.queryParent&&(this.parentMap=new WeakMap)),this.evtListeners={}}parse(g,C,I){I=I?I+".":"";let A=/<([^ >\/]+)(.*?)>/gm,l=null,b=[];for(;l=A.exec(g);){let c=l[1],o={$tag:c},Z=I+c,d=l[2].trim(),G=!1;(d.endsWith("/")||c.startsWith("?")||c.startsWith("!"))&&(G=!0);let s=/([^ ]+?)="(.+?)"/g,e=/([^ ]+?)='(.+?)'/g,n=null,t=!1;for(;n=s.exec(d);)t=!0,o[n[1]]=n[2];if(!t)for(;n=e.exec(d);)t=!0,o[n[1]]=n[2];if(t||""===d||(o.text=d),this.progressive&&this.emit(`<${Z}>`,o,C),!G){let C=new RegExp(`([^]+?)</${c}>`,"g");C.lastIndex=A.lastIndex;let I=C.exec(g);if(I&&I[1]){A.lastIndex=C.lastIndex;let g=this.parse(I[1],o,Z);g.length>0?o.$innerNodes=g:o.$innerText=I[1]}}this.queryParent&&C&&this.parentMap.set(o,C),this.progressive&&this.emit(`</${Z}>`,o,C),b.push(o)}return b}getParent(g){return this.queryParent?this.parentMap.get(g):null}$addListener(g,C){let I=this.evtListeners[g];I?I.push(C):this.evtListeners[g]=[C]}addListener(I,A){g(I)&&(I=C(I),A.condition=function(g){let C="return "+g.replace(/(\$.+?)(?=[=!.])/g,"node.$&")+";";return new Function("node",C)}(I.exp),I=I.evt),this.$addListener(I,A)}$removeListener(g,C){let I=this.evtListeners[g],A=null;I&&(A=I.indexOf(C))>=0&&I.splice(A,1)}removeListener(I,A){g(I)&&(I=(I=C(I)).evt),this.$removeListener(I,A)}emit(g,...C){let I=this.evtListeners[g];if(I)for(let g of I)g.condition?!0===g.condition.apply(null,C)&&g.apply(null,C):g.apply(null,C)}on(g,C){this.addListener(g,C)}off(g,C){this.removeListener(g,C)}}})();const{Node:Ug,Way:Mg,Relation:Og}=Qg,{purgeProps:jg,RefElements:Dg}=fg,Pg=Tg;var Eg=(g,C)=>{let I=!1,A=!1,l=!0;(g=>{if(g){I=!(!g.completeFeature&&!g.allFeatures),A=!!g.renderTagged;let C=g.suppressWay||g.excludeWay;void 0===C||C||(l=!1)}})(C);let b=(g=>g.elements?"json":g.indexOf("<osm")>=0?"xml":g.trim().startsWith("{")?"json-raw":"invalid")(g),c=new Dg,o=[];"json-raw"===b&&(b=(g=JSON.parse(g)).elements?"json":"invalid"),"json"===b?(g=>{for(let C of g.elements)switch(C.type){case"node":let g=new Ug(C.id,c);C.tags&&g.addTags(C.tags),g.addProps(jg(C,["id","type","tags","lat","lon"])),g.setLatLng(C);break;case"way":let I=new Mg(C.id,c);if(C.tags&&I.addTags(C.tags),I.addProps(jg(C,["id","type","tags","nodes","geometry"])),C.nodes)for(let g of C.nodes)I.addNodeRef(g);else C.geometry&&I.setLatLngArray(C.geometry);break;case"relation":let A=new Og(C.id,c);if(C.bounds&&A.setBounds([parseFloat(C.bounds.minlon),parseFloat(C.bounds.minlat),parseFloat(C.bounds.maxlon),parseFloat(C.bounds.maxlat)]),C.tags&&A.addTags(C.tags),A.addProps(jg(C,["id","type","tags","bounds","members"])),C.members)for(let g of C.members)A.addMember(g)}})(g):"xml"===b&&(g=>{const C=new Pg({progressive:!0});C.on("</osm.node>",(g=>{let C=new Ug(g.id,c);for(let[I,A]of Object.entries(g))!I.startsWith("$")&&["id","lon","lat"].indexOf(I)<0&&C.addProp(I,A);if(C.setLatLng(g),g.$innerNodes)for(let I of g.$innerNodes)"tag"===I.$tag&&C.addTag(I.k,I.v)})),C.on("</osm.way>",(g=>{let C=new Mg(g.id,c);for(let[I,A]of Object.entries(g))!I.startsWith("$")&&["id"].indexOf(I)<0&&C.addProp(I,A);if(g.$innerNodes)for(let I of g.$innerNodes)"nd"===I.$tag?I.lon&&I.lat?C.addLatLng(I):I.ref&&C.addNodeRef(I.ref):"tag"===I.$tag&&C.addTag(I.k,I.v)})),C.on("<osm.relation>",(g=>{new Og(g.id,c)})),C.on("</osm.relation.member>",((g,C)=>{let I=c.get(`relation/${C.id}`),A={type:g.type,role:g.role?g.role:"",ref:g.ref};if(g.lat&&g.lon){A.lat=g.lat,A.lon=g.lon,A.tags={};for(let[C,I]of Object.entries(g))!C.startsWith("$")&&["type","lat","lon"].indexOf(C)<0&&(A[C]=I)}if(g.$innerNodes){let C=[],I=[];for(let A of g.$innerNodes)A.lat&&A.lon?C.push(A):I.push(A.ref);C.length>0?A.geometry=C:I.length>0&&(A.nodes=I)}I.addMember(A)})),C.on("</osm.relation.bounds>",((g,C)=>{c.get(`relation/${C.id}`).setBounds([parseFloat(g.minlon),parseFloat(g.minlat),parseFloat(g.maxlon),parseFloat(g.maxlat)])})),C.on("</osm.relation.tag>",((g,C)=>{c.get(`relation/${C.id}`).addTag(g.k,g.v)})),C.parse(g)})(g),c.bindAll();for(let g of c.values())if(g.refCount<=0||g.hasTag&&A&&!(g instanceof Mg&&l)){let C=g.toFeatureArray();if(g instanceof Og&&!I&&C.length>0)return C[0].geometry;o=o.concat(C)}return{type:"FeatureCollection",features:o}},qg=C(Eg);const _g=["topojson","osm","kml","gpx","tcx","csv","tsv"];class $g{constructor(g,C){this.blankGeoJSON=()=>({type:"FeatureCollection",features:[]}),this._rawData=C,this._format=g;const I={topojson:this.loadTopoJson,osm:this.loadOsm,kml:this.loadXml,gpx:this.loadXml,tcx:this.loadXml,csv:this.loadCsv,tsv:this.loadCsv};this._conversionFn=I[g]}async convert(){return this._conversionFn?this._conversionFn():new Promise(((g,C)=>C(`No converter exists for ${this._format}`)))}async loadXml(){return Lg[this._format]((new DOMParser).parseFromString(this._rawData,"text/xml"))}async loadCsv(){let g={};"tsv"===this._format&&(g.delimiter="\t");return await new Promise(((C,I)=>{H.csv2geojson(this._rawData,g,((g,A)=>{g?I(g):C(A)}))}))}async loadTopoJson(){let g={};try{g=JSON.parse(this._rawData)}catch(g){throw"Invalid TopoJson"}let C=this.blankGeoJSON();return"Topology"===g.type&&void 0!==g.objects&&(C={type:"FeatureCollection",features:C.features=Object.keys(g.objects).map((C=>{return I=g,"string"==typeof(A=C)&&(A=I.objects[A]),"GeometryCollection"===A.type?{type:"FeatureCollection",features:A.geometries.map((function(g){return R(I,g)}))}:R(I,A);var I,A})).reduce(((g,C)=>[...g,...C.features]),[])}),C}async loadOsm(){return qg(this._rawData)}}var gC=null;try{var CC="undefined"!=typeof module&&"function"==typeof module.require&&module.require("worker_threads")||"function"==typeof __non_webpack_require__&&__non_webpack_require__("worker_threads")||"function"==typeof require&&require("worker_threads");gC=CC.Worker}catch(g){}function IC(g,C,I){var A=void 0===C?null:C,l=function(g,C){return Buffer.from(g,"base64").toString(C?"utf16":"utf8")}(g,void 0!==I&&I),b=l.indexOf("\n",10)+1,c=l.substring(b)+(A?"//# sourceMappingURL="+A:"");return function(g){return new gC(c,Object.assign({},g,{eval:!0}))}}function AC(g,C,I){var A=void 0===C?null:C,l=function(g,C){var I=atob(g);if(C){for(var A=new Uint8Array(I.length),l=0,b=I.length;l<b;++l)A[l]=I.charCodeAt(l);return String.fromCharCode.apply(null,new Uint16Array(A.buffer))}return I}(g,void 0!==I&&I),b=l.indexOf("\n",10)+1,c=l.substring(b)+(A?"//# sourceMappingURL="+A:""),o=new Blob([c],{type:"application/javascript"});return URL.createObjectURL(o)}var lC="[object process]"===Object.prototype.toString.call("undefined"!=typeof process?process:0);var bC,cC,oC,ZC=(bC="/* rollup-plugin-web-worker-loader */
(function () {
	'use strict';

	function getDefaultExportFromCjs (x) {
		return x && x.__esModule && Object.prototype.hasOwnProperty.call(x, 'default') ? x['default'] : x;
	}

	function getAugmentedNamespace(n) {
	  if (n.__esModule) return n;
	  var f = n.default;
		if (typeof f == "function") {
			var a = function a () {
				if (this instanceof a) {
					var args = [null];
					args.push.apply(args, arguments);
					var Ctor = Function.bind.apply(f, args);
					return new Ctor();
				}
				return f.apply(this, arguments);
			};
			a.prototype = f.prototype;
	  } else a = {};
	  Object.defineProperty(a, '__esModule', {value: true});
		Object.keys(n).forEach(function (k) {
			var d = Object.getOwnPropertyDescriptor(n, k);
			Object.defineProperty(a, k, d.get ? d : {
				enumerable: true,
				get: function () {
					return n[k];
				}
			});
		});
		return a;
	}

	function objectConverter(columns) {
	  return new Function("d", "return {" + columns.map(function(name, i) {
	    return JSON.stringify(name) + ": d[" + i + "]";
	  }).join(",") + "}");
	}

	function customConverter(columns, f) {
	  var object = objectConverter(columns);
	  return function(row, i) {
	    return f(object(row), i, columns);
	  };
	}

	// Compute unique columns in order of discovery.
	function inferColumns(rows) {
	  var columnSet = Object.create(null),
	      columns = [];

	  rows.forEach(function(row) {
	    for (var column in row) {
	      if (!(column in columnSet)) {
	        columns.push(columnSet[column] = column);
	      }
	    }
	  });

	  return columns;
	}

	function dsv$1(delimiter) {
	  var reFormat = new RegExp("[\"" + delimiter + "\n]"),
	      delimiterCode = delimiter.charCodeAt(0);

	  function parse(text, f) {
	    var convert, columns, rows = parseRows(text, function(row, i) {
	      if (convert) return convert(row, i - 1);
	      columns = row, convert = f ? customConverter(row, f) : objectConverter(row);
	    });
	    rows.columns = columns;
	    return rows;
	  }

	  function parseRows(text, f) {
	    var EOL = {}, // sentinel value for end-of-line
	        EOF = {}, // sentinel value for end-of-file
	        rows = [], // output rows
	        N = text.length,
	        I = 0, // current character index
	        n = 0, // the current line number
	        t, // the current token
	        eol; // is the current token followed by EOL?

	    function token() {
	      if (I >= N) return EOF; // special case: end of file
	      if (eol) return eol = false, EOL; // special case: end of line

	      // special case: quotes
	      var j = I, c;
	      if (text.charCodeAt(j) === 34) {
	        var i = j;
	        while (i++ < N) {
	          if (text.charCodeAt(i) === 34) {
	            if (text.charCodeAt(i + 1) !== 34) break;
	            ++i;
	          }
	        }
	        I = i + 2;
	        c = text.charCodeAt(i + 1);
	        if (c === 13) {
	          eol = true;
	          if (text.charCodeAt(i + 2) === 10) ++I;
	        } else if (c === 10) {
	          eol = true;
	        }
	        return text.slice(j + 1, i).replace(/""/g, "\"");
	      }

	      // common case: find next delimiter or newline
	      while (I < N) {
	        var k = 1;
	        c = text.charCodeAt(I++);
	        if (c === 10) eol = true; // \n
	        else if (c === 13) { eol = true; if (text.charCodeAt(I) === 10) ++I, ++k; } // \r|\r\n
	        else if (c !== delimiterCode) continue;
	        return text.slice(j, I - k);
	      }

	      // special case: last token before EOF
	      return text.slice(j);
	    }

	    while ((t = token()) !== EOF) {
	      var a = [];
	      while (t !== EOL && t !== EOF) {
	        a.push(t);
	        t = token();
	      }
	      if (f && (a = f(a, n++)) == null) continue;
	      rows.push(a);
	    }

	    return rows;
	  }

	  function format(rows, columns) {
	    if (columns == null) columns = inferColumns(rows);
	    return [columns.map(formatValue).join(delimiter)].concat(rows.map(function(row) {
	      return columns.map(function(column) {
	        return formatValue(row[column]);
	      }).join(delimiter);
	    })).join("\n");
	  }

	  function formatRows(rows) {
	    return rows.map(formatRow).join("\n");
	  }

	  function formatRow(row) {
	    return row.map(formatValue).join(delimiter);
	  }

	  function formatValue(text) {
	    return text == null ? ""
	        : reFormat.test(text += "") ? "\"" + text.replace(/\"/g, "\"\"") + "\""
	        : text;
	  }

	  return {
	    parse: parse,
	    parseRows: parseRows,
	    format: format,
	    formatRows: formatRows
	  };
	}

	var csv = dsv$1(",");

	var csvParse = csv.parse;
	var csvParseRows = csv.parseRows;
	var csvFormat = csv.format;
	var csvFormatRows = csv.formatRows;

	var tsv = dsv$1("\t");

	var tsvParse = tsv.parse;
	var tsvParseRows = tsv.parseRows;
	var tsvFormat = tsv.format;
	var tsvFormatRows = tsv.formatRows;

	var d3Dsv = /*#__PURE__*/Object.freeze({
		__proto__: null,
		dsvFormat: dsv$1,
		csvParse: csvParse,
		csvParseRows: csvParseRows,
		csvFormat: csvFormat,
		csvFormatRows: csvFormatRows,
		tsvParse: tsvParse,
		tsvParseRows: tsvParseRows,
		tsvFormat: tsvFormat,
		tsvFormatRows: tsvFormatRows
	});

	var require$$0 = /*@__PURE__*/getAugmentedNamespace(d3Dsv);

	var sexagesimal$1 = {exports: {}};

	sexagesimal$1.exports = element;
	sexagesimal$1.exports.pair = pair;
	sexagesimal$1.exports.format = format;
	sexagesimal$1.exports.formatPair = formatPair;
	sexagesimal$1.exports.coordToDMS = coordToDMS;


	function element(input, dims) {
	  var result = search(input, dims);
	  return (result === null) ? null : result.val;
	}


	function formatPair(input) {
	  return format(input.lat, 'lat') + ' ' + format(input.lon, 'lon');
	}


	// Is 0 North or South?
	function format(input, dim) {
	  var dms = coordToDMS(input, dim);
	  return dms.whole + '° ' +
	    (dms.minutes ? dms.minutes + '\' ' : '') +
	    (dms.seconds ? dms.seconds + '" ' : '') + dms.dir;
	}


	function coordToDMS(input, dim) {
	  var dirs = { lat: ['N', 'S'], lon: ['E', 'W'] }[dim] || '';
	  var dir = dirs[input >= 0 ? 0 : 1];
	  var abs = Math.abs(input);
	  var whole = Math.floor(abs);
	  var fraction = abs - whole;
	  var fractionMinutes = fraction * 60;
	  var minutes = Math.floor(fractionMinutes);
	  var seconds = Math.floor((fractionMinutes - minutes) * 60);

	  return {
	    whole: whole,
	    minutes: minutes,
	    seconds: seconds,
	    dir: dir
	  };
	}


	function search(input, dims) {
	  if (!dims) dims = 'NSEW';
	  if (typeof input !== 'string') return null;

	  input = input.toUpperCase();
	  var regex = /^[\s\,]*([NSEW])?\s*([\-|\—|\―]?[0-9.]+)[°º˚]?\s*(?:([0-9.]+)['’′‘]\s*)?(?:([0-9.]+)(?:''|"|”|″)\s*)?([NSEW])?/;

	  var m = input.match(regex);
	  if (!m) return null;  // no match

	  var matched = m[0];

	  // extract dimension.. m[1] = leading, m[5] = trailing
	  var dim;
	  if (m[1] && m[5]) {                 // if matched both..
	    dim = m[1];                       // keep leading
	    matched = matched.slice(0, -1);   // remove trailing dimension from match
	  } else {
	    dim = m[1] || m[5];
	  }

	  // if unrecognized dimension
	  if (dim && dims.indexOf(dim) === -1) return null;

	  // extract DMS
	  var deg = m[2] ? parseFloat(m[2]) : 0;
	  var min = m[3] ? parseFloat(m[3]) / 60 : 0;
	  var sec = m[4] ? parseFloat(m[4]) / 3600 : 0;
	  var sign = (deg < 0) ? -1 : 1;
	  if (dim === 'S' || dim === 'W') sign *= -1;

	  return {
	    val: (Math.abs(deg) + min + sec) * sign,
	    dim: dim,
	    matched: matched,
	    remain: input.slice(matched.length)
	  };
	}


	function pair(input, dims) {
	  input = input.trim();
	  var one = search(input, dims);
	  if (!one) return null;

	  input = one.remain.trim();
	  var two = search(input, dims);
	  if (!two || two.remain) return null;

	  if (one.dim) {
	    return swapdim(one.val, two.val, one.dim);
	  } else {
	    return [one.val, two.val];
	  }
	}


	function swapdim(a, b, dim) {
	  if (dim === 'N' || dim === 'S') return [a, b];
	  if (dim === 'W' || dim === 'E') return [b, a];
	}

	var sexagesimalExports = sexagesimal$1.exports;

	var dsv = require$$0,
	    sexagesimal = sexagesimalExports;

	var latRegex = /(Lat)(itude)?/gi,
	    lonRegex = /(L)(on|ng)(gitude)?/i;

	function guessHeader(row, regexp) {
	    var name, match, score;
	    for (var f in row) {
	        match = f.match(regexp);
	        if (match && (!name || match[0].length / f.length > score)) {
	            score = match[0].length / f.length;
	            name = f;
	        }
	    }
	    return name;
	}

	function guessLatHeader(row) { return guessHeader(row, latRegex); }
	function guessLonHeader(row) { return guessHeader(row, lonRegex); }

	function isLat(f) { return !!f.match(latRegex); }
	function isLon(f) { return !!f.match(lonRegex); }

	function keyCount(o) {
	    return (typeof o == 'object') ? Object.keys(o).length : 0;
	}

	function autoDelimiter(x) {
	    var delimiters = [',', ';', '\t', '|'];
	    var results = [];

	    delimiters.forEach(function (delimiter) {
	        var res = dsv.dsvFormat(delimiter).parse(x);
	        if (res.length >= 1) {
	            var count = keyCount(res[0]);
	            for (var i = 0; i < res.length; i++) {
	                if (keyCount(res[i]) !== count) return;
	            }
	            results.push({
	                delimiter: delimiter,
	                arity: Object.keys(res[0]).length,
	            });
	        }
	    });

	    if (results.length) {
	        return results.sort(function (a, b) {
	            return b.arity - a.arity;
	        })[0].delimiter;
	    } else {
	        return null;
	    }
	}

	/**
	 * Silly stopgap for dsv to d3-dsv upgrade
	 *
	 * @param {Array} x dsv output
	 * @returns {Array} array without columns member
	 */
	function deleteColumns(x) {
	    delete x.columns;
	    return x;
	}

	function auto(x) {
	    var delimiter = autoDelimiter(x);
	    if (!delimiter) return null;
	    return deleteColumns(dsv.dsvFormat(delimiter).parse(x));
	}

	function csv2geojson(x, options, callback) {

	    if (!callback) {
	        callback = options;
	        options = {};
	    }

	    options.delimiter = options.delimiter || ',';

	    var latfield = options.latfield || '',
	        lonfield = options.lonfield || '',
	        crs = options.crs || '';

	    var features = [],
	        featurecollection = {type: 'FeatureCollection', features: features};

	    if (crs !== '') {
	        featurecollection.crs = {type: 'name', properties: {name: crs}};
	    }

	    if (options.delimiter === 'auto' && typeof x == 'string') {
	        options.delimiter = autoDelimiter(x);
	        if (!options.delimiter) {
	            callback({
	                type: 'Error',
	                message: 'Could not autodetect delimiter'
	            });
	            return;
	        }
	    }

	    var numericFields = options.numericFields ? options.numericFields.split(',') : null;

	    var parsed = (typeof x == 'string') ?
	        dsv.dsvFormat(options.delimiter).parse(x, function (d) {
	            if (numericFields) {
	                for (var key in d) {
	                    if (numericFields.includes(key)) {
	                        d[key] = +d[key];
	                    }
	                }
	            }
	            return d;
	        }) : x;

	    if (!parsed.length) {
	        callback(null, featurecollection);
	        return;
	    }

	    var errors = [];
	    var i;


	    if (!latfield) latfield = guessLatHeader(parsed[0]);
	    if (!lonfield) lonfield = guessLonHeader(parsed[0]);
	    var noGeometry = (!latfield || !lonfield);

	    if (noGeometry) {
	        for (i = 0; i < parsed.length; i++) {
	            features.push({
	                type: 'Feature',
	                properties: parsed[i],
	                geometry: null
	            });
	        }
	        callback(errors.length ? errors : null, featurecollection);
	        return;
	    }

	    for (i = 0; i < parsed.length; i++) {
	        if (parsed[i][lonfield] !== undefined &&
	            parsed[i][latfield] !== undefined) {

	            var lonk = parsed[i][lonfield],
	                latk = parsed[i][latfield],
	                lonf, latf,
	                a;

	            a = sexagesimal(lonk, 'EW');
	            if (a) lonk = a;
	            a = sexagesimal(latk, 'NS');
	            if (a) latk = a;

	            lonf = parseFloat(lonk);
	            latf = parseFloat(latk);

	            if (isNaN(lonf) ||
	                isNaN(latf)) {
	                errors.push({
	                    message: 'A row contained an invalid value for latitude or longitude',
	                    row: parsed[i],
	                    index: i
	                });
	            } else {
	                if (!options.includeLatLon) {
	                    delete parsed[i][lonfield];
	                    delete parsed[i][latfield];
	                }

	                features.push({
	                    type: 'Feature',
	                    properties: parsed[i],
	                    geometry: {
	                        type: 'Point',
	                        coordinates: [
	                            parseFloat(lonf),
	                            parseFloat(latf)
	                        ]
	                    }
	                });
	            }
	        }
	    }

	    callback(errors.length ? errors : null, featurecollection);
	}

	function toLine(gj) {
	    var features = gj.features;
	    var line = {
	        type: 'Feature',
	        geometry: {
	            type: 'LineString',
	            coordinates: []
	        }
	    };
	    for (var i = 0; i < features.length; i++) {
	        line.geometry.coordinates.push(features[i].geometry.coordinates);
	    }
	    line.properties = features.reduce(function (aggregatedProperties, newFeature) {
	        for (var key in newFeature.properties) {
	            if (!aggregatedProperties[key]) {
	                aggregatedProperties[key] = [];
	            }
	            aggregatedProperties[key].push(newFeature.properties[key]);
	        }
	        return aggregatedProperties;
	    }, {});
	    return {
	        type: 'FeatureCollection',
	        features: [line]
	    };
	}

	function toPolygon(gj) {
	    var features = gj.features;
	    var poly = {
	        type: 'Feature',
	        geometry: {
	            type: 'Polygon',
	            coordinates: [[]]
	        }
	    };
	    for (var i = 0; i < features.length; i++) {
	        poly.geometry.coordinates[0].push(features[i].geometry.coordinates);
	    }
	    poly.properties = features.reduce(function (aggregatedProperties, newFeature) {
	        for (var key in newFeature.properties) {
	            if (!aggregatedProperties[key]) {
	                aggregatedProperties[key] = [];
	            }
	            aggregatedProperties[key].push(newFeature.properties[key]);
	        }
	        return aggregatedProperties;
	    }, {});
	    return {
	        type: 'FeatureCollection',
	        features: [poly]
	    };
	}

	var csv2geojson_1 = {
	    isLon: isLon,
	    isLat: isLat,
	    guessLatHeader: guessLatHeader,
	    guessLonHeader: guessLonHeader,
	    csv: dsv.csvParse,
	    tsv: dsv.tsvParse,
	    dsv: dsv,
	    auto: auto,
	    csv2geojson: csv2geojson,
	    toLine: toLine,
	    toPolygon: toPolygon
	};

	function identity(x) {
	  return x;
	}

	function transform(transform) {
	  if (transform == null) return identity;
	  var x0,
	      y0,
	      kx = transform.scale[0],
	      ky = transform.scale[1],
	      dx = transform.translate[0],
	      dy = transform.translate[1];
	  return function(input, i) {
	    if (!i) x0 = y0 = 0;
	    var j = 2, n = input.length, output = new Array(n);
	    output[0] = (x0 += input[0]) * kx + dx;
	    output[1] = (y0 += input[1]) * ky + dy;
	    while (j < n) output[j] = input[j], ++j;
	    return output;
	  };
	}

	function reverse(array, n) {
	  var t, j = array.length, i = j - n;
	  while (i < --j) t = array[i], array[i++] = array[j], array[j] = t;
	}

	function topojsonFeature(topology, o) {
	  if (typeof o === "string") o = topology.objects[o];
	  return o.type === "GeometryCollection"
	      ? {type: "FeatureCollection", features: o.geometries.map(function(o) { return feature(topology, o); })}
	      : feature(topology, o);
	}

	function feature(topology, o) {
	  var id = o.id,
	      bbox = o.bbox,
	      properties = o.properties == null ? {} : o.properties,
	      geometry = object(topology, o);
	  return id == null && bbox == null ? {type: "Feature", properties: properties, geometry: geometry}
	      : bbox == null ? {type: "Feature", id: id, properties: properties, geometry: geometry}
	      : {type: "Feature", id: id, bbox: bbox, properties: properties, geometry: geometry};
	}

	function object(topology, o) {
	  var transformPoint = transform(topology.transform),
	      arcs = topology.arcs;

	  function arc(i, points) {
	    if (points.length) points.pop();
	    for (var a = arcs[i < 0 ? ~i : i], k = 0, n = a.length; k < n; ++k) {
	      points.push(transformPoint(a[k], k));
	    }
	    if (i < 0) reverse(points, n);
	  }

	  function point(p) {
	    return transformPoint(p);
	  }

	  function line(arcs) {
	    var points = [];
	    for (var i = 0, n = arcs.length; i < n; ++i) arc(arcs[i], points);
	    if (points.length < 2) points.push(points[0]); // This should never happen per the specification.
	    return points;
	  }

	  function ring(arcs) {
	    var points = line(arcs);
	    while (points.length < 4) points.push(points[0]); // This may happen if an arc has only two points.
	    return points;
	  }

	  function polygon(arcs) {
	    return arcs.map(ring);
	  }

	  function geometry(o) {
	    var type = o.type, coordinates;
	    switch (type) {
	      case "GeometryCollection": return {type: type, geometries: o.geometries.map(geometry)};
	      case "Point": coordinates = point(o.coordinates); break;
	      case "MultiPoint": coordinates = o.coordinates.map(point); break;
	      case "LineString": coordinates = line(o.arcs); break;
	      case "MultiLineString": coordinates = o.arcs.map(line); break;
	      case "Polygon": coordinates = polygon(o.arcs); break;
	      case "MultiPolygon": coordinates = o.arcs.map(polygon); break;
	      default: return null;
	    }
	    return {type: type, coordinates: coordinates};
	  }

	  return geometry(o);
	}

	function $(element, tagName) {
	    return Array.from(element.getElementsByTagName(tagName));
	}
	function normalizeId(id) {
	    return id[0] === "#" ? id : `#${id}`;
	}
	function $ns(element, tagName, ns) {
	    return Array.from(element.getElementsByTagNameNS(ns, tagName));
	}
	/**
	 * get the content of a text node, if any
	 */
	function nodeVal(node) {
	    node?.normalize();
	    return (node && node.textContent) || "";
	}
	/**
	 * Get one Y child of X, if any, otherwise null
	 */
	function get1(node, tagName, callback) {
	    const n = node.getElementsByTagName(tagName);
	    const result = n.length ? n[0] : null;
	    if (result && callback)
	        callback(result);
	    return result;
	}
	function get(node, tagName, callback) {
	    const properties = {};
	    if (!node)
	        return properties;
	    const n = node.getElementsByTagName(tagName);
	    const result = n.length ? n[0] : null;
	    if (result && callback) {
	        return callback(result, properties);
	    }
	    return properties;
	}
	function val1(node, tagName, callback) {
	    const val = nodeVal(get1(node, tagName));
	    if (val && callback)
	        return callback(val) || {};
	    return {};
	}
	function $num(node, tagName, callback) {
	    const val = parseFloat(nodeVal(get1(node, tagName)));
	    if (isNaN(val))
	        return undefined;
	    if (val && callback)
	        return callback(val) || {};
	    return {};
	}
	function num1(node, tagName, callback) {
	    const val = parseFloat(nodeVal(get1(node, tagName)));
	    if (isNaN(val))
	        return undefined;
	    if (val && callback)
	        callback(val);
	    return val;
	}
	function getMulti(node, propertyNames) {
	    const properties = {};
	    for (const property of propertyNames) {
	        val1(node, property, (val) => {
	            properties[property] = val;
	        });
	    }
	    return properties;
	}
	function isElement(node) {
	    return node?.nodeType === 1;
	}

	function getLineStyle(node) {
	    return get(node, "line", (lineStyle) => {
	        const val = Object.assign({}, val1(lineStyle, "color", (color) => {
	            return { stroke: `#${color}` };
	        }), $num(lineStyle, "opacity", (opacity) => {
	            return { "stroke-opacity": opacity };
	        }), $num(lineStyle, "width", (width) => {
	            // GPX width is in mm, convert to px with 96 px per inch
	            return { "stroke-width": (width * 96) / 25.4 };
	        }));
	        return val;
	    });
	}

	function getExtensions(node) {
	    let values = [];
	    if (node === null)
	        return values;
	    for (const child of Array.from(node.childNodes)) {
	        if (!isElement(child))
	            continue;
	        const name = abbreviateName(child.nodeName);
	        if (name === "gpxtpx:TrackPointExtension") {
	            // loop again for nested garmin extensions (eg. "gpxtpx:hr")
	            values = values.concat(getExtensions(child));
	        }
	        else {
	            // push custom extension (eg. "power")
	            const val = nodeVal(child);
	            values.push([name, parseNumeric(val)]);
	        }
	    }
	    return values;
	}
	function abbreviateName(name) {
	    return ["heart", "gpxtpx:hr", "hr"].includes(name) ? "heart" : name;
	}
	function parseNumeric(val) {
	    const num = parseFloat(val);
	    return isNaN(num) ? val : num;
	}

	function coordPair$1(node) {
	    const ll = [
	        parseFloat(node.getAttribute("lon") || ""),
	        parseFloat(node.getAttribute("lat") || ""),
	    ];
	    if (isNaN(ll[0]) || isNaN(ll[1])) {
	        return null;
	    }
	    num1(node, "ele", (val) => {
	        ll.push(val);
	    });
	    const time = get1(node, "time");
	    return {
	        coordinates: ll,
	        time: time ? nodeVal(time) : null,
	        extendedValues: getExtensions(get1(node, "extensions")),
	    };
	}

	function extractProperties(node) {
	    const properties = getMulti(node, [
	        "name",
	        "cmt",
	        "desc",
	        "type",
	        "time",
	        "keywords",
	    ]);
	    const extensions = Array.from(node.getElementsByTagNameNS("http://www.garmin.com/xmlschemas/GpxExtensions/v3", "*"));
	    for (const child of extensions) {
	        if (child.parentNode?.parentNode === node) {
	            properties[child.tagName.replace(":", "_")] = nodeVal(child);
	        }
	    }
	    const links = $(node, "link");
	    if (links.length) {
	        properties.links = links.map((link) => Object.assign({ href: link.getAttribute("href") }, getMulti(link, ["text", "type"])));
	    }
	    return properties;
	}

	/**
	 * Extract points from a trkseg or rte element.
	 */
	function getPoints$1(node, pointname) {
	    const pts = $(node, pointname);
	    const line = [];
	    const times = [];
	    const extendedValues = {};
	    for (let i = 0; i < pts.length; i++) {
	        const c = coordPair$1(pts[i]);
	        if (!c) {
	            continue;
	        }
	        line.push(c.coordinates);
	        if (c.time)
	            times.push(c.time);
	        for (const [name, val] of c.extendedValues) {
	            const plural = name === "heart" ? name : name.replace("gpxtpx:", "") + "s";
	            if (!extendedValues[plural]) {
	                extendedValues[plural] = Array(pts.length).fill(null);
	            }
	            extendedValues[plural][i] = val;
	        }
	    }
	    if (line.length < 2)
	        return; // Invalid line in GeoJSON
	    return {
	        line: line,
	        times: times,
	        extendedValues: extendedValues,
	    };
	}
	/**
	 * Extract a LineString geometry from a rte
	 * element.
	 */
	function getRoute(node) {
	    const line = getPoints$1(node, "rtept");
	    if (!line)
	        return;
	    return {
	        type: "Feature",
	        properties: Object.assign({ _gpxType: "rte" }, extractProperties(node), getLineStyle(get1(node, "extensions"))),
	        geometry: {
	            type: "LineString",
	            coordinates: line.line,
	        },
	    };
	}
	function getTrack(node) {
	    const segments = $(node, "trkseg");
	    const track = [];
	    const times = [];
	    const extractedLines = [];
	    for (const segment of segments) {
	        const line = getPoints$1(segment, "trkpt");
	        if (line) {
	            extractedLines.push(line);
	            if (line.times && line.times.length)
	                times.push(line.times);
	        }
	    }
	    if (extractedLines.length === 0)
	        return null;
	    const multi = extractedLines.length > 1;
	    const properties = Object.assign({ _gpxType: "trk" }, extractProperties(node), getLineStyle(get1(node, "extensions")), times.length
	        ? {
	            coordinateProperties: {
	                times: multi ? times : times[0],
	            },
	        }
	        : {});
	    for (const line of extractedLines) {
	        track.push(line.line);
	        if (!properties.coordinateProperties) {
	            properties.coordinateProperties = {};
	        }
	        const props = properties.coordinateProperties;
	        const entries = Object.entries(line.extendedValues);
	        for (let i = 0; i < entries.length; i++) {
	            const [name, val] = entries[i];
	            if (multi) {
	                if (!props[name]) {
	                    props[name] = extractedLines.map((line) => new Array(line.line.length).fill(null));
	                }
	                props[name][i] = val;
	            }
	            else {
	                props[name] = val;
	            }
	        }
	    }
	    return {
	        type: "Feature",
	        properties: properties,
	        geometry: multi
	            ? {
	                type: "MultiLineString",
	                coordinates: track,
	            }
	            : {
	                type: "LineString",
	                coordinates: track[0],
	            },
	    };
	}
	/**
	 * Extract a point, if possible, from a given node,
	 * which is usually a wpt or trkpt
	 */
	function getPoint(node) {
	    const properties = Object.assign(extractProperties(node), getMulti(node, ["sym"]));
	    const pair = coordPair$1(node);
	    if (!pair)
	        return null;
	    return {
	        type: "Feature",
	        properties,
	        geometry: {
	            type: "Point",
	            coordinates: pair.coordinates,
	        },
	    };
	}
	/**
	 * Convert GPX to GeoJSON incrementally, returning
	 * a [Generator](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Iterators_and_Generators)
	 * that yields output feature by feature.
	 */
	function* gpxGen(node) {
	    for (const track of $(node, "trk")) {
	        const feature = getTrack(track);
	        if (feature)
	            yield feature;
	    }
	    for (const route of $(node, "rte")) {
	        const feature = getRoute(route);
	        if (feature)
	            yield feature;
	    }
	    for (const waypoint of $(node, "wpt")) {
	        const point = getPoint(waypoint);
	        if (point)
	            yield point;
	    }
	}
	/**
	 *
	 * Convert a GPX document to GeoJSON. The first argument, `doc`, must be a GPX
	 * document as an XML DOM - not as a string. You can get this using jQuery's default
	 * `.ajax` function or using a bare XMLHttpRequest with the `.response` property
	 * holding an XML DOM.
	 *
	 * The output is a JavaScript object of GeoJSON data, same as `.kml` outputs, with the
	 * addition of a `_gpxType` property on each `LineString` feature that indicates whether
	 * the feature was encoded as a route (`rte`) or track (`trk`) in the GPX document.
	 */
	function gpx(node) {
	    return {
	        type: "FeatureCollection",
	        features: Array.from(gpxGen(node)),
	    };
	}

	const EXTENSIONS_NS = "http://www.garmin.com/xmlschemas/ActivityExtension/v2";
	const TRACKPOINT_ATTRIBUTES = [
	    ["heartRate", "heartRates"],
	    ["Cadence", "cadences"],
	    // Extended Trackpoint attributes
	    ["Speed", "speeds"],
	    ["Watts", "watts"],
	];
	const LAP_ATTRIBUTES = [
	    ["TotalTimeSeconds", "totalTimeSeconds"],
	    ["DistanceMeters", "distanceMeters"],
	    ["MaximumSpeed", "maxSpeed"],
	    ["AverageHeartRateBpm", "avgHeartRate"],
	    ["MaximumHeartRateBpm", "maxHeartRate"],
	    // Extended Lap attributes
	    ["AvgSpeed", "avgSpeed"],
	    ["AvgWatts", "avgWatts"],
	    ["MaxWatts", "maxWatts"],
	];
	function getProperties(node, attributeNames) {
	    const properties = [];
	    for (const [tag, alias] of attributeNames) {
	        let elem = get1(node, tag);
	        if (!elem) {
	            const elements = node.getElementsByTagNameNS(EXTENSIONS_NS, tag);
	            if (elements.length) {
	                elem = elements[0];
	            }
	        }
	        const val = parseFloat(nodeVal(elem));
	        if (!isNaN(val)) {
	            properties.push([alias, val]);
	        }
	    }
	    return properties;
	}
	function coordPair(node) {
	    const ll = [num1(node, "LongitudeDegrees"), num1(node, "LatitudeDegrees")];
	    if (ll[0] === undefined ||
	        isNaN(ll[0]) ||
	        ll[1] === undefined ||
	        isNaN(ll[1])) {
	        return null;
	    }
	    const heartRate = get1(node, "HeartRateBpm");
	    const time = nodeVal(get1(node, "Time"));
	    get1(node, "AltitudeMeters", (alt) => {
	        const a = parseFloat(nodeVal(alt));
	        if (!isNaN(a)) {
	            ll.push(a);
	        }
	    });
	    return {
	        coordinates: ll,
	        time: time || null,
	        heartRate: heartRate ? parseFloat(nodeVal(heartRate)) : null,
	        extensions: getProperties(node, TRACKPOINT_ATTRIBUTES),
	    };
	}
	function getPoints(node) {
	    const pts = $(node, "Trackpoint");
	    const line = [];
	    const times = [];
	    const heartRates = [];
	    if (pts.length < 2)
	        return null; // Invalid line in GeoJSON
	    const extendedProperties = {};
	    const result = { extendedProperties };
	    for (let i = 0; i < pts.length; i++) {
	        const c = coordPair(pts[i]);
	        if (c === null)
	            continue;
	        line.push(c.coordinates);
	        const { time, heartRate, extensions } = c;
	        if (time)
	            times.push(time);
	        if (heartRate)
	            heartRates.push(heartRate);
	        for (const [alias, value] of extensions) {
	            if (!extendedProperties[alias]) {
	                extendedProperties[alias] = Array(pts.length).fill(null);
	            }
	            extendedProperties[alias][i] = value;
	        }
	    }
	    if (line.length < 2)
	        return null;
	    return Object.assign(result, {
	        line: line,
	        times: times,
	        heartRates: heartRates,
	    });
	}
	function getLap(node) {
	    const segments = $(node, "Track");
	    const track = [];
	    const times = [];
	    const heartRates = [];
	    const allExtendedProperties = [];
	    let line;
	    const properties = Object.assign(Object.fromEntries(getProperties(node, LAP_ATTRIBUTES)), get(node, "Name", (nameElement) => {
	        return { name: nodeVal(nameElement) };
	    }));
	    for (const segment of segments) {
	        line = getPoints(segment);
	        if (line) {
	            track.push(line.line);
	            if (line.times.length)
	                times.push(line.times);
	            if (line.heartRates.length)
	                heartRates.push(line.heartRates);
	            allExtendedProperties.push(line.extendedProperties);
	        }
	    }
	    for (let i = 0; i < allExtendedProperties.length; i++) {
	        const extendedProperties = allExtendedProperties[i];
	        for (const property in extendedProperties) {
	            if (segments.length === 1) {
	                if (line) {
	                    properties[property] = line.extendedProperties[property];
	                }
	            }
	            else {
	                if (!properties[property]) {
	                    properties[property] = track.map((track) => Array(track.length).fill(null));
	                }
	                properties[property][i] = extendedProperties[property];
	            }
	        }
	    }
	    if (track.length === 0)
	        return null;
	    if (times.length || heartRates.length) {
	        properties.coordinateProperties = Object.assign(times.length
	            ? {
	                times: track.length === 1 ? times[0] : times,
	            }
	            : {}, heartRates.length
	            ? {
	                heart: track.length === 1 ? heartRates[0] : heartRates,
	            }
	            : {});
	    }
	    return {
	        type: "Feature",
	        properties: properties,
	        geometry: track.length === 1
	            ? {
	                type: "LineString",
	                coordinates: track[0],
	            }
	            : {
	                type: "MultiLineString",
	                coordinates: track,
	            },
	    };
	}
	/**
	 * Incrementally convert a TCX document to GeoJSON. The
	 * first argument, `doc`, must be a TCX
	 * document as an XML DOM - not as a string.
	 */
	function* tcxGen(node) {
	    for (const lap of $(node, "Lap")) {
	        const feature = getLap(lap);
	        if (feature)
	            yield feature;
	    }
	    for (const course of $(node, "Courses")) {
	        const feature = getLap(course);
	        if (feature)
	            yield feature;
	    }
	}
	/**
	 * Convert a TCX document to GeoJSON. The first argument, `doc`, must be a TCX
	 * document as an XML DOM - not as a string.
	 */
	function tcx(node) {
	    return {
	        type: "FeatureCollection",
	        features: Array.from(tcxGen(node)),
	    };
	}

	function fixColor(v, prefix) {
	    const properties = {};
	    const colorProp = prefix == "stroke" || prefix === "fill" ? prefix : prefix + "-color";
	    if (v[0] === "#") {
	        v = v.substring(1);
	    }
	    if (v.length === 6 || v.length === 3) {
	        properties[colorProp] = "#" + v;
	    }
	    else if (v.length === 8) {
	        properties[prefix + "-opacity"] = parseInt(v.substring(0, 2), 16) / 255;
	        properties[colorProp] =
	            "#" + v.substring(6, 8) + v.substring(4, 6) + v.substring(2, 4);
	    }
	    return properties;
	}

	function numericProperty(node, source, target) {
	    const properties = {};
	    num1(node, source, (val) => {
	        properties[target] = val;
	    });
	    return properties;
	}
	function getColor(node, output) {
	    return get(node, "color", (elem) => fixColor(nodeVal(elem), output));
	}
	function extractIconHref(node) {
	    return get(node, "Icon", (icon, properties) => {
	        val1(icon, "href", (href) => {
	            properties.icon = href;
	        });
	        return properties;
	    });
	}
	function extractIcon(node) {
	    return get(node, "IconStyle", (iconStyle) => {
	        return Object.assign(getColor(iconStyle, "icon"), numericProperty(iconStyle, "scale", "icon-scale"), numericProperty(iconStyle, "heading", "icon-heading"), get(iconStyle, "hotSpot", (hotspot) => {
	            const left = parseFloat(hotspot.getAttribute("x") || "");
	            const top = parseFloat(hotspot.getAttribute("y") || "");
	            const xunits = hotspot.getAttribute("xunits") || "";
	            const yunits = hotspot.getAttribute("yunits") || "";
	            if (!isNaN(left) && !isNaN(top))
	                return {
	                    "icon-offset": [left, top],
	                    "icon-offset-units": [xunits, yunits],
	                };
	            return {};
	        }), extractIconHref(iconStyle));
	    });
	}
	function extractLabel(node) {
	    return get(node, "LabelStyle", (labelStyle) => {
	        return Object.assign(getColor(labelStyle, "label"), numericProperty(labelStyle, "scale", "label-scale"));
	    });
	}
	function extractLine(node) {
	    return get(node, "LineStyle", (lineStyle) => {
	        return Object.assign(getColor(lineStyle, "stroke"), numericProperty(lineStyle, "width", "stroke-width"));
	    });
	}
	function extractPoly(node) {
	    return get(node, "PolyStyle", (polyStyle, properties) => {
	        return Object.assign(properties, get(polyStyle, "color", (elem) => fixColor(nodeVal(elem), "fill")), val1(polyStyle, "fill", (fill) => {
	            if (fill === "0")
	                return { "fill-opacity": 0 };
	        }), val1(polyStyle, "outline", (outline) => {
	            if (outline === "0")
	                return { "stroke-opacity": 0 };
	        }));
	    });
	}
	function extractStyle(node) {
	    return Object.assign({}, extractPoly(node), extractLine(node), extractLabel(node), extractIcon(node));
	}

	const toNumber = (x) => Number(x);
	const typeConverters = {
	    string: (x) => x,
	    int: toNumber,
	    uint: toNumber,
	    short: toNumber,
	    ushort: toNumber,
	    float: toNumber,
	    double: toNumber,
	    bool: (x) => Boolean(x),
	};
	function extractExtendedData(node, schema) {
	    return get(node, "ExtendedData", (extendedData, properties) => {
	        for (const data of $(extendedData, "Data")) {
	            properties[data.getAttribute("name") || ""] = nodeVal(get1(data, "value"));
	        }
	        for (const simpleData of $(extendedData, "SimpleData")) {
	            const name = simpleData.getAttribute("name") || "";
	            const typeConverter = schema[name] || typeConverters.string;
	            properties[name] = typeConverter(nodeVal(simpleData));
	        }
	        return properties;
	    });
	}
	function getMaybeHTMLDescription(node) {
	    const descriptionNode = get1(node, "description");
	    for (const c of Array.from(descriptionNode?.childNodes || [])) {
	        if (c.nodeType === 4) {
	            return {
	                description: {
	                    "@type": "html",
	                    value: nodeVal(c),
	                },
	            };
	        }
	    }
	    return {};
	}
	function extractTimeSpan(node) {
	    return get(node, "TimeSpan", (timeSpan) => {
	        return {
	            timespan: {
	                begin: nodeVal(get1(timeSpan, "begin")),
	                end: nodeVal(get1(timeSpan, "end")),
	            },
	        };
	    });
	}
	function extractTimeStamp(node) {
	    return get(node, "TimeStamp", (timeStamp) => {
	        return { timestamp: nodeVal(get1(timeStamp, "when")) };
	    });
	}
	function extractCascadedStyle(node, styleMap) {
	    return val1(node, "styleUrl", (styleUrl) => {
	        styleUrl = normalizeId(styleUrl);
	        if (styleMap[styleUrl]) {
	            return Object.assign({ styleUrl }, styleMap[styleUrl]);
	        }
	        // For backward-compatibility. Should we still include
	        // styleUrl even if it's not resolved?
	        return { styleUrl };
	    });
	}

	const removeSpace = /\s*/g;
	const trimSpace = /^\s*|\s*$/g;
	const splitSpace = /\s+/;
	/**
	 * Get one coordinate from a coordinate array, if any
	 */
	function coord1(value) {
	    return value
	        .replace(removeSpace, "")
	        .split(",")
	        .map(parseFloat)
	        .filter((num) => !isNaN(num))
	        .slice(0, 3);
	}
	/**
	 * Get all coordinates from a coordinate array as [[],[]]
	 */
	function coord(value) {
	    return value
	        .replace(trimSpace, "")
	        .split(splitSpace)
	        .map(coord1)
	        .filter((coord) => {
	        return coord.length >= 2;
	    });
	}
	function gxCoords(node) {
	    let elems = $(node, "coord");
	    if (elems.length === 0) {
	        elems = $ns(node, "coord", "*");
	    }
	    const coordinates = elems.map((elem) => {
	        return nodeVal(elem).split(" ").map(parseFloat);
	    });
	    if (coordinates.length === 0) {
	        return null;
	    }
	    return {
	        geometry: coordinates.length > 2
	            ? {
	                type: "LineString",
	                coordinates,
	            }
	            : {
	                type: "Point",
	                coordinates: coordinates[0],
	            },
	        times: $(node, "when").map((elem) => nodeVal(elem)),
	    };
	}
	function fixRing(ring) {
	    if (ring.length === 0)
	        return ring;
	    const first = ring[0];
	    const last = ring[ring.length - 1];
	    let equal = true;
	    for (let i = 0; i < Math.max(first.length, last.length); i++) {
	        if (first[i] !== last[i]) {
	            equal = false;
	            break;
	        }
	    }
	    if (!equal) {
	        return ring.concat([ring[0]]);
	    }
	    return ring;
	}
	function getCoordinates(node) {
	    return nodeVal(get1(node, "coordinates"));
	}
	function getGeometry(node) {
	    let geometries = [];
	    let coordTimes = [];
	    for (let i = 0; i < node.childNodes.length; i++) {
	        const child = node.childNodes.item(i);
	        if (isElement(child)) {
	            switch (child.tagName) {
	                case "MultiGeometry":
	                case "MultiTrack":
	                case "gx:MultiTrack": {
	                    const childGeometries = getGeometry(child);
	                    geometries = geometries.concat(childGeometries.geometries);
	                    coordTimes = coordTimes.concat(childGeometries.coordTimes);
	                    break;
	                }
	                case "Point": {
	                    const coordinates = coord1(getCoordinates(child));
	                    if (coordinates.length >= 2) {
	                        geometries.push({
	                            type: "Point",
	                            coordinates,
	                        });
	                    }
	                    break;
	                }
	                case "LinearRing":
	                case "LineString": {
	                    const coordinates = coord(getCoordinates(child));
	                    if (coordinates.length >= 2) {
	                        geometries.push({
	                            type: "LineString",
	                            coordinates,
	                        });
	                    }
	                    break;
	                }
	                case "Polygon": {
	                    const coords = [];
	                    for (const linearRing of $(child, "LinearRing")) {
	                        const ring = fixRing(coord(getCoordinates(linearRing)));
	                        if (ring.length >= 4) {
	                            coords.push(ring);
	                        }
	                    }
	                    if (coords.length) {
	                        geometries.push({
	                            type: "Polygon",
	                            coordinates: coords,
	                        });
	                    }
	                    break;
	                }
	                case "Track":
	                case "gx:Track": {
	                    const gx = gxCoords(child);
	                    if (!gx)
	                        break;
	                    const { times, geometry } = gx;
	                    geometries.push(geometry);
	                    if (times.length)
	                        coordTimes.push(times);
	                    break;
	                }
	            }
	        }
	    }
	    return {
	        geometries,
	        coordTimes,
	    };
	}

	function geometryListToGeometry(geometries) {
	    return geometries.length === 0
	        ? null
	        : geometries.length === 1
	            ? geometries[0]
	            : {
	                type: "GeometryCollection",
	                geometries,
	            };
	}
	function getPlacemark(node, styleMap, schema) {
	    const { coordTimes, geometries } = getGeometry(node);
	    const feature = {
	        type: "Feature",
	        geometry: geometryListToGeometry(geometries),
	        properties: Object.assign(getMulti(node, [
	            "name",
	            "address",
	            "visibility",
	            "open",
	            "phoneNumber",
	            "description",
	        ]), getMaybeHTMLDescription(node), extractCascadedStyle(node, styleMap), extractStyle(node), extractExtendedData(node, schema), extractTimeSpan(node), extractTimeStamp(node), coordTimes.length
	            ? {
	                coordinateProperties: {
	                    times: coordTimes.length === 1 ? coordTimes[0] : coordTimes,
	                },
	            }
	            : {}),
	    };
	    if (feature.properties?.visibility !== undefined) {
	        feature.properties.visibility = feature.properties.visibility !== "0";
	    }
	    const id = node.getAttribute("id");
	    if (id !== null && id !== "")
	        feature.id = id;
	    return feature;
	}

	function getGroundOverlayBox(node) {
	    const latLonQuad = get1(node, "gx:LatLonQuad");
	    if (latLonQuad) {
	        const ring = fixRing(coord(getCoordinates(node)));
	        return {
	            type: "Polygon",
	            coordinates: [ring],
	        };
	    }
	    return getLatLonBox(node);
	}
	const DEGREES_TO_RADIANS = Math.PI / 180;
	function rotateBox(bbox, coordinates, rotation) {
	    const center = [(bbox[0] + bbox[2]) / 2, (bbox[1] + bbox[3]) / 2];
	    return [
	        coordinates[0].map((coordinate) => {
	            const dy = coordinate[1] - center[1];
	            const dx = coordinate[0] - center[0];
	            const distance = Math.sqrt(Math.pow(dy, 2) + Math.pow(dx, 2));
	            const angle = Math.atan2(dy, dx) - rotation * DEGREES_TO_RADIANS;
	            return [
	                center[0] + Math.cos(angle) * distance,
	                center[1] + Math.sin(angle) * distance,
	            ];
	        }),
	    ];
	}
	function getLatLonBox(node) {
	    const latLonBox = get1(node, "LatLonBox");
	    if (latLonBox) {
	        const north = num1(latLonBox, "north");
	        const west = num1(latLonBox, "west");
	        const east = num1(latLonBox, "east");
	        const south = num1(latLonBox, "south");
	        const rotation = num1(latLonBox, "rotation");
	        if (typeof north === "number" &&
	            typeof south === "number" &&
	            typeof west === "number" &&
	            typeof east === "number") {
	            const bbox = [west, south, east, north];
	            let coordinates = [
	                [
	                    [west, north],
	                    [east, north],
	                    [east, south],
	                    [west, south],
	                    [west, north], // top left (again)
	                ],
	            ];
	            if (typeof rotation === "number") {
	                coordinates = rotateBox(bbox, coordinates, rotation);
	            }
	            return {
	                type: "Polygon",
	                coordinates,
	            };
	        }
	    }
	    return null;
	}
	function getGroundOverlay(node, styleMap, schema) {
	    const geometry = getGroundOverlayBox(node);
	    const feature = {
	        type: "Feature",
	        geometry,
	        properties: Object.assign(
	        /**
	         * Related to
	         * https://gist.github.com/tmcw/037a1cb6660d74a392e9da7446540f46
	         */
	        { "@geometry-type": "groundoverlay" }, getMulti(node, [
	            "name",
	            "address",
	            "visibility",
	            "open",
	            "phoneNumber",
	            "description",
	        ]), getMaybeHTMLDescription(node), extractCascadedStyle(node, styleMap), extractStyle(node), extractIconHref(node), extractExtendedData(node, schema), extractTimeSpan(node), extractTimeStamp(node)),
	    };
	    if (feature.properties?.visibility !== undefined) {
	        feature.properties.visibility = feature.properties.visibility !== "0";
	    }
	    const id = node.getAttribute("id");
	    if (id !== null && id !== "")
	        feature.id = id;
	    return feature;
	}

	function getStyleId(style) {
	    let id = style.getAttribute("id");
	    const parentNode = style.parentNode;
	    if (!id &&
	        isElement(parentNode) &&
	        parentNode.localName === "CascadingStyle") {
	        id = parentNode.getAttribute("kml:id") || parentNode.getAttribute("id");
	    }
	    return normalizeId(id || "");
	}
	function buildStyleMap(node) {
	    const styleMap = {};
	    for (const style of $(node, "Style")) {
	        styleMap[getStyleId(style)] = extractStyle(style);
	    }
	    for (const map of $(node, "StyleMap")) {
	        const id = normalizeId(map.getAttribute("id") || "");
	        val1(map, "styleUrl", (styleUrl) => {
	            styleUrl = normalizeId(styleUrl);
	            if (styleMap[styleUrl]) {
	                styleMap[id] = styleMap[styleUrl];
	            }
	        });
	    }
	    return styleMap;
	}
	function buildSchema(node) {
	    const schema = {};
	    for (const field of $(node, "SimpleField")) {
	        schema[field.getAttribute("name") || ""] =
	            typeConverters[field.getAttribute("type") || ""] ||
	                typeConverters["string"];
	    }
	    return schema;
	}
	const FOLDER_PROPS = [
	    "name",
	    "visibility",
	    "open",
	    "address",
	    "description",
	    "phoneNumber",
	    "visibility",
	];
	function getFolder(node) {
	    const meta = {};
	    for (const child of Array.from(node.childNodes)) {
	        if (isElement(child) && FOLDER_PROPS.includes(child.tagName)) {
	            meta[child.tagName] = nodeVal(child);
	        }
	    }
	    return {
	        type: "folder",
	        meta,
	        children: [],
	    };
	}
	/**
	 * Yield a nested tree with KML folder structure
	 *
	 * This generates a tree with the given structure:
	 *
	 * ```js
	 * {
	 *   "type": "root",
	 *   "children": [
	 *     {
	 *       "type": "folder",
	 *       "meta": {
	 *         "name": "Test"
	 *       },
	 *       "children": [
	 *          // ...features and folders
	 *       ]
	 *     }
	 *     // ...features
	 *   ]
	 * }
	 * ```
	 *
	 * ### GroundOverlay
	 *
	 * GroundOverlay elements are converted into
	 * `Feature` objects with `Polygon` geometries,
	 * a property like:
	 *
	 * ```json
	 * {
	 *   "@geometry-type": "groundoverlay"
	 * }
	 * ```
	 *
	 * And the ground overlay's image URL in the `href`
	 * property. Ground overlays will need to be displayed
	 * with a separate method to other features, depending
	 * on which map framework you're using.
	 */
	function kmlWithFolders(node) {
	    const styleMap = buildStyleMap(node);
	    const schema = buildSchema(node);
	    const tree = { type: "root", children: [] };
	    function traverse(node, pointer) {
	        if (isElement(node)) {
	            switch (node.tagName) {
	                case "GroundOverlay": {
	                    const placemark = getGroundOverlay(node, styleMap, schema);
	                    if (placemark) {
	                        pointer.children.push(placemark);
	                    }
	                    break;
	                }
	                case "Placemark": {
	                    const placemark = getPlacemark(node, styleMap, schema);
	                    if (placemark) {
	                        pointer.children.push(placemark);
	                    }
	                    break;
	                }
	                case "Folder": {
	                    const folder = getFolder(node);
	                    pointer.children.push(folder);
	                    pointer = folder;
	                    break;
	                }
	            }
	        }
	        if (node.childNodes) {
	            for (let i = 0; i < node.childNodes.length; i++) {
	                traverse(node.childNodes[i], pointer);
	            }
	        }
	    }
	    traverse(node, tree);
	    return tree;
	}
	/**
	 * Convert KML to GeoJSON incrementally, returning
	 * a [Generator](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Iterators_and_Generators)
	 * that yields output feature by feature.
	 */
	function* kmlGen(node) {
	    const styleMap = buildStyleMap(node);
	    const schema = buildSchema(node);
	    for (const placemark of $(node, "Placemark")) {
	        const feature = getPlacemark(placemark, styleMap, schema);
	        if (feature)
	            yield feature;
	    }
	    for (const groundOverlay of $(node, "GroundOverlay")) {
	        const feature = getGroundOverlay(groundOverlay, styleMap, schema);
	        if (feature)
	            yield feature;
	    }
	}
	/**
	 * Convert a KML document to GeoJSON. The first argument, `doc`, must be a KML
	 * document as an XML DOM - not as a string. You can get this using jQuery's default
	 * `.ajax` function or using a bare XMLHttpRequest with the `.response` property
	 * holding an XML DOM.
	 *
	 * The output is a JavaScript object of GeoJSON data. You can convert it to a string
	 * with [JSON.stringify](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/JSON/stringify)
	 * or use it directly in libraries.
	 */
	function kml(node) {
	    return {
	        type: "FeatureCollection",
	        features: Array.from(kmlGen(node)),
	    };
	}

	var toGeoJson = /*#__PURE__*/Object.freeze({
		__proto__: null,
		gpx: gpx,
		gpxGen: gpxGen,
		kml: kml,
		kmlGen: kmlGen,
		kmlWithFolders: kmlWithFolders,
		tcx: tcx,
		tcxGen: tcxGen
	});

	var utils = (() => {

	    const purgeProps = (obj, blacklist) => {
	        if (obj) {
	            let rs = Object.assign({}, obj);
	            if (blacklist) {
	                for (let prop of blacklist) {
	                    delete rs[prop];
	                }
	            }
	            return rs;
	        }
	        return {};
	    };

	    const mergeProps = (obj1, obj2) => {
	        obj1 = obj1? obj1 : {};
	        obj2 = obj2? obj2 : {};
	        return Object.assign(obj1, obj2);
	    };

	    const first = a => a[0];
	    const last = a => a[a.length - 1];
	    const coordsToKey = a => a.join(',');

	    const addToMap = (m, k, v) => {
	        let a = m[k];
	        if (a) {
	            a.push(v);
	        } else {
	            m[k] = [v];
	        }
	    };
	    
	    const removeFromMap = (m, k, v) => {
	        let a = m[k];
	        let idx = null;
	        if (a && (idx = a.indexOf(v)) >= 0) {
	            a.splice(idx, 1);
	        }
	    };
	    
	    const getFirstFromMap = (m, k) => {
	        let a = m[k];
	        if (a && a.length > 0) {
	            return a[0];
	        }
	        return null;
	    };

	    // need 3+ different points to form a ring, here using > 3 is 'coz a the first and the last points are actually the same
	    const isRing = a => a.length > 3 && coordsToKey(first(a)) === coordsToKey(last(a));

	    const ringDirection = (a, xIdx, yIdx) => {
	        xIdx = xIdx || 0, yIdx = yIdx || 1;
	        // get the index of the point which has the maximum x value
	        let m = a.reduce((maxxIdx, v, idx) => a[maxxIdx][xIdx] > v[xIdx] ? maxxIdx : idx, 0);
	        // 'coz the first point is virtually the same one as the last point, 
	        // we need to skip a.length - 1 for left when m = 0,
	        // and skip 0 for right when m = a.length - 1;
	        let l = m <= 0? a.length - 2 : m - 1, r = m >= a.length - 1? 1 : m + 1;
	        let xa = a[l][xIdx], xb = a[m][xIdx], xc = a[r][xIdx];
	        let ya = a[l][yIdx], yb = a[m][yIdx], yc = a[r][yIdx];
	        let det = (xb - xa) * (yc - ya) - (xc - xa) * (yb - ya);
	        return det < 0 ? 'clockwise' : 'counterclockwise';
	    };

	    const ptInsidePolygon = (pt, polygon, xIdx, yIdx) => {
	        xIdx = xIdx || 0, yIdx = yIdx || 1;
	        let result = false;
	        for (let i = 0, j = polygon.length - 1; i < polygon.length; j = i++) {
	            if ((polygon[i][xIdx] <= pt[xIdx] && pt[xIdx] < polygon[j][xIdx] ||
	                polygon[j][xIdx] <= pt[xIdx] && pt[xIdx] < polygon[i][xIdx]) &&
	                pt[yIdx] < (polygon[j][yIdx] - polygon[i][yIdx]) * (pt[xIdx] - polygon[i][xIdx]) / (polygon[j][xIdx] - polygon[i][xIdx]) + polygon[i][yIdx]) {
	                    result = !result;
	                }
	                
	        }
	        return result;
	    };

	    const strToFloat = el => el instanceof Array? el.map(strToFloat) : parseFloat(el);
	    
	    class RefElements extends Map {
	        constructor() {
	            super();
	            this.binders = [];
	        }

	        add(k, v) {
	            if (!this.has(k)) {
	                this.set(k, v);
	            }
	            // suppress duplcated key error
	            // else
	            // throw `Error: adding duplicated key '${k}' to RefElements`;
	        }

	        addBinder(binder) {
	            this.binders.push(binder);
	        }

	        bindAll() {
	            this.binders.forEach(binder => binder.bind());
	        }
	    }

	    class LateBinder {
	        constructor(container, valueFunc, ctx, args) {
	            this.container = container;
	            this.valueFunc = valueFunc;
	            this.ctx = ctx;
	            this.args = args;
	        }

	        bind() {
	            let v = this.valueFunc.apply(this.ctx, this.args);
	            if (this.container instanceof Array) {
	                let idx = this.container.indexOf(this);
	                if (idx >= 0) {
	                    let args = [idx, 1];
	                    if (v) {
	                        args.push(v);
	                    }
	                    [].splice.apply(this.container, args);
	                }
	            } else if (typeof this.container === 'object') {
	                let k = Object.keys(this.container).find(v => this.container[v] === this);
	                if (k) {
	                    if (v) {
	                        this.container[k] = v;
	                    } else {
	                        delete this.container[k];
	                    }
	                }
	            }
	        }
	    }

	    class WayCollection extends Array {
	        constructor() {
	            super();
	            this.firstMap = {};
	            this.lastMap = {};
	        }

	        addWay(way) {
	            way = way.toCoordsArray();
	            if (way.length > 0) {
	                this.push(way);
	                addToMap(this.firstMap, coordsToKey(first(way)), way);
	                addToMap(this.lastMap, coordsToKey(last(way)), way);
	            }
	        }

	        toStrings() {
	            let strings = [], way = null;
	            while (way = this.shift()) {
	                removeFromMap(this.firstMap, coordsToKey(first(way)), way);
	                removeFromMap(this.lastMap, coordsToKey(last(way)), way);
	                let current = way, next = null;
	                do {
	                    let key = coordsToKey(last(current)), shouldReverse = false;

	                    next = getFirstFromMap(this.firstMap, key);                                        
	                    if (!next) {
	                        next = getFirstFromMap(this.lastMap, key);
	                        shouldReverse = true;
	                    }
	                    
	                    if (next) {
	                        this.splice(this.indexOf(next), 1);
	                        removeFromMap(this.firstMap, coordsToKey(first(next)), next);
	                        removeFromMap(this.lastMap, coordsToKey(last(next)), next);
	                        if (shouldReverse) {
	                            // always reverse shorter one to save time
	                            if (next.length > current.length) {
	                                [current, next] = [next, current];
	                            }
	                            next.reverse();
	                        }

	                        current = current.concat(next.slice(1));
	                    }
	                } while (next);
	                strings.push(strToFloat(current));
	            }

	            return strings;
	        }

	        toRings(direction) {
	            let strings = this.toStrings();
	            let rings = [], string = null;
	            while (string = strings.shift()) {
	                if (isRing(string)) {
	                    if (ringDirection(string) !== direction) {
	                        string.reverse();
	                    }
	                    rings.push(string);
	                }    
	            }
	            return rings;
	        }
	    }

	    return {purgeProps, mergeProps,
	        first, last, coordsToKey,
	        addToMap, removeFromMap, getFirstFromMap,
	        isRing, ringDirection, ptInsidePolygon, strToFloat,
	        RefElements, LateBinder, WayCollection};
	})();

	var building = {
	};
	var highway = {
		whitelist: [
			"services",
			"rest_area",
			"escape",
			"elevator"
		]
	};
	var natural = {
		blacklist: [
			"coastline",
			"cliff",
			"ridge",
			"arete",
			"tree_row"
		]
	};
	var landuse = {
	};
	var waterway = {
		whitelist: [
			"riverbank",
			"dock",
			"boatyard",
			"dam"
		]
	};
	var amenity = {
	};
	var leisure = {
	};
	var barrier = {
		whitelist: [
			"city_wall",
			"ditch",
			"hedge",
			"retaining_wall",
			"wall",
			"spikes"
		]
	};
	var railway = {
		whitelist: [
			"station",
			"turntable",
			"roundhouse",
			"platform"
		]
	};
	var area = {
	};
	var boundary = {
	};
	var man_made = {
		blacklist: [
			"cutline",
			"embankment",
			"pipeline"
		]
	};
	var power = {
		whitelist: [
			"plant",
			"substation",
			"generator",
			"transformer"
		]
	};
	var place = {
	};
	var shop = {
	};
	var aeroway = {
		blacklist: [
			"taxiway"
		]
	};
	var tourism = {
	};
	var historic = {
	};
	var public_transport = {
	};
	var office = {
	};
	var military = {
	};
	var ruins = {
	};
	var craft = {
	};
	var golf = {
	};
	var indoor = {
	};
	var require$$1 = {
		building: building,
		highway: highway,
		natural: natural,
		landuse: landuse,
		waterway: waterway,
		amenity: amenity,
		leisure: leisure,
		barrier: barrier,
		railway: railway,
		area: area,
		boundary: boundary,
		man_made: man_made,
		power: power,
		place: place,
		shop: shop,
		aeroway: aeroway,
		tourism: tourism,
		historic: historic,
		public_transport: public_transport,
		office: office,
		"building:part": {
	},
		military: military,
		ruins: ruins,
		"area:highway": {
	},
		craft: craft,
		golf: golf,
		indoor: indoor
	};

	var osmobjs = (() => {

	    const {first, last, coordsToKey,
	        addToMap, removeFromMap, getFirstFromMap, 
	        isRing, ringDirection, ptInsidePolygon, strToFloat, 
	        LateBinder, WayCollection} = utils,
	        polygonTags = require$$1;

	    class OsmObject {
	        constructor(type, id, refElems) {
	            this.type = type;
	            this.id = id;
	            this.refElems = refElems;
	            this.tags = {};
	            this.props = {id: this.getCompositeId()};
	            this.refCount = 0;
	            this.hasTag = false;
	            if (refElems) {
	                refElems.add(this.getCompositeId(), this);
	            }
	        }

	        addTags(tags) {
	            this.tags = Object.assign(this.tags, tags);
	            this.hasTag = tags? true : false;
	        }

	        addTag(k, v) {
	            this.tags[k] = v;
	            this.hasTag = k? true : false;
	        }

	        addProp(k, v) {
	            this.props[k] = v;
	        }

	        addProps(props) {
	            this.props = Object.assign(this.props, props);    
	        }

	        getCompositeId() {
	            return `${this.type}/${this.id}`;
	        }

	        getProps() {
	            return Object.assign(this.props, this.tags);
	        }        

	        toFeatureArray() {
	            return [];
	        }
	    }

	    class Node extends OsmObject {
	        constructor(id, refElems) {
	            super('node', id, refElems);
	            this.latLng = null;
	        }

	        setLatLng(latLng) {
	            this.latLng = latLng;
	        }

	        toFeatureArray() {
	            if (this.latLng) {
	                return [{
	                    type: 'Feature',
	                    id: this.getCompositeId(),
	                    properties: this.getProps(),
	                    geometry: {
	                        type: 'Point',
	                        coordinates: strToFloat([this.latLng.lon, this.latLng.lat])
	                    }
	                }];
	            }

	            return [];
	        }

	        getLatLng() {
	            return this.latLng;
	        }
	    }

	    class Way extends OsmObject {
	        constructor(id, refElems) {
	            super('way', id, refElems);
	            this.latLngArray = [];
	            this.isPolygon = false;
	        }

	        addLatLng(latLng) {
	            this.latLngArray.push(latLng);
	        }

	        setLatLngArray(latLngArray) {
	            this.latLngArray = latLngArray;
	        }

	        addNodeRef(ref) {
	            let binder = new LateBinder(this.latLngArray, function(id) {
	                let node = this.refElems.get(`node/${id}`);
	                if (node) {
	                    node.refCount++;
	                    return node.getLatLng();
	                }
	            }, this, [ref]);

	            this.latLngArray.push(binder);
	            this.refElems.addBinder(binder);
	        }

	        analyzeTag(k, v) {
	            let o = polygonTags[k];
	            if (o) {
	                this.isPolygon = true;
	                if (o.whitelist) {
	                    this.isPolygon = o.whitelist.indexOf(v) >= 0? true : false;
	                } else if(o.blacklist) {
	                    this.isPolygon = o.blacklist.indexOf(v) >= 0? false : true;
	                }
	            }
	        }

	        addTags(tags) {
	            super.addTags(tags);
	            for (let [k, v] of Object.entries(tags)) {
	                this.analyzeTag(k, v);
	            }
	        }

	        addTag(k, v) {
	            super.addTag(k, v);
	            this.analyzeTag(k, v);
	        }

	        toCoordsArray() {
	            return this.latLngArray.map(latLng => [latLng.lon, latLng.lat]);
	        }

	        toFeatureArray() {
	            let coordsArray = this.toCoordsArray();
	            if (coordsArray.length > 1) {
	                coordsArray = strToFloat(coordsArray);
	                let feature = {
	                    type: 'Feature',
	                    id: this.getCompositeId(),
	                    properties: this.getProps(),
	                    geometry: {
	                        type: 'LineString',
	                        coordinates: coordsArray
	                    }
	                };

	                if (this.isPolygon && isRing(coordsArray)) {
	                    if (ringDirection(coordsArray) !== 'counterclockwise') {
	                        coordsArray.reverse();
	                    }

	                    feature.geometry = {
	                        type: 'Polygon',
	                        coordinates: [coordsArray]
	                    };

	                    return [feature];
	                }

	                return [feature];
	            }

	            return [];
	        }
	    }

	    class Relation extends OsmObject {
	        constructor(id, refElems) {
	            super('relation', id, refElems);
	            this.relations = [];
	            this.nodes = [];
	            this.bounds = null;
	        }

	        setBounds(bounds) {
	            this.bounds = bounds;
	        }

	        addMember(member) {
	            switch (member.type) {
	                // super relation, need to do combination
	                case 'relation':
	                    let binder = new LateBinder(this.relations, function(id) {
	                        let relation = this.refElems.get(`relation/${id}`);
	                        if (relation) {
	                            relation.refCount++;
	                            return relation;
	                        }
	                    }, this, [member.ref]);
	                    this.relations.push(binder);
	                    this.refElems.addBinder(binder);
	                    break;

	                case 'way':
	                    if (!member.role) {
	                        member.role = '';
	                    }
	                    let ways = this[member.role];
	                    if (!ways) {
	                        ways = this[member.role] = [];
	                    }
	                    if (member.geometry) {
	                        let way = new Way(member.ref, this.refElems);
	                        way.setLatLngArray(member.geometry);
	                        way.refCount++;
	                        ways.push(way);
	                    } else if (member.nodes) {
	                        let way = new Way(member.ref, this.refElems);
	                        for (let nid of member.nodes) {
	                            way.addNodeRef(nid);
	                        }
	                        way.refCount++;
	                        ways.push(way);
	                    } else {
	                        let binder = new LateBinder(ways, function(id) {
	                            let way = this.refElems.get(`way/${id}`);
	                            if (way) {
	                                way.refCount++;
	                                return way;
	                            }
	                        }, this, [member.ref]);
	                        ways.push(binder);
	                        this.refElems.addBinder(binder);
	                    }
	                    break;

	                case 'node':
	                    let node = null;
	                    if (member.lat && member.lon) {
	                        node = new Node(member.ref, this.refElems);
	                        node.setLatLng({lon: member.lon, lat: member.lat});
	                        if (member.tags) {
	                            node.addTags(member.tags);
	                        }
	                        for (let [k, v] of Object.entries(member)) {
	                            if (['id', 'type', 'lat', 'lon'].indexOf(k) < 0) {
	                                node.addProp(k, v);
	                            }
	                        }

	                        node.refCount++;
	                        this.nodes.push(node);
	                    } else {
	                        let binder = new LateBinder(this.nodes, function(id) {
	                            let node = this.refElems.get(`node/${id}`);
	                            if (node) {
	                                node.refCount++;
	                                return node;
	                            }
	                        }, this, [member.ref]);
	                        this.nodes.push(binder);
	                        this.refElems.addBinder(binder);
	                    }
	                    break;
	            }
	        }

	        toFeatureArray() {
	            const constructStringGeometry = (ws) => {
	                let strings = ws? ws.toStrings() : [];
	                if (strings.length > 0) {
	                    if (strings.length === 1) return {
	                        type: 'LineString',
	                        coordinates: strings[0]
	                    }

	                    return {
	                        type: 'MultiLineString',
	                        coordinates: strings
	                    }
	                }
	                return null;
	            };

	            const constructPolygonGeometry = (ows, iws) => {
	                let outerRings = ows? ows.toRings('counterclockwise') : [],
	                    innerRings = iws? iws.toRings('clockwise') : [];
	                                
	                if (outerRings.length > 0) {
	                    let compositPolyons = [];

	                    let ring = null;
	                    for (ring of outerRings)
	                        compositPolyons.push([ring]);
	                    
	                    // link inner polygons to outer containers
	                    while (ring = innerRings.shift()) {
	                        for (let idx in outerRings) {
	                            if (ptInsidePolygon(first(ring), outerRings[idx])) {
	                                compositPolyons[idx].push(ring);
	                                break;
	                            }
	                        }
	                    }

	                    // construct the Polygon/MultiPolygon geometry
	                    if (compositPolyons.length === 1) {
	                        return {
	                            type: 'Polygon',
	                            coordinates: compositPolyons[0]
	                        };
	                    }

	                    return {
	                        type: 'MultiPolygon',
	                        coordinates: compositPolyons
	                    }
	                }

	                return null;
	            };

	            let polygonFeatures = [], stringFeatures = [], pointFeatures = [];
	            const waysFieldNames = ['outer', 'inner', ''];
	            // need to do combination when there're nested relations
	            for (let relation of this.relations) {
	                if (relation) {
	                    for (let fieldName of waysFieldNames) {
	                        let ways = relation[fieldName];
	                        if (ways) {
	                            let thisWays = this[fieldName];
	                            if (thisWays) {
	                                [].splice.apply(thisWays, [thisWays.length, 0].concat(ways));
	                            } else {
	                                this[fieldName] = ways;
	                            }
	                        }
	                    }
	                }
	            }

	            for (let fieldName of waysFieldNames) {
	                let ways = this[fieldName];
	                if (ways) {
	                    this[fieldName] = new WayCollection();
	                    for (let way of ways) {
	                        this[fieldName].addWay(way);
	                    }
	                }
	            }

	            let geometry = null;
	            
	            let feature = {
	                type: 'Feature',
	                id: this.getCompositeId(),
	                bbox: this.bounds,
	                properties: this.getProps()
	            };

	            if (!this.bounds) {
	                delete feature.bbox;
	            }

	            if (this.outer) {
	                geometry = constructPolygonGeometry(this.outer, this.inner);
	                if (geometry) {
	                    feature.geometry = geometry;
	                    polygonFeatures.push(feature);
	                }
	            }
	            else if (this['']) {
	                geometry = constructStringGeometry(this['']);
	                if (geometry) {
	                    feature.geometry = geometry;
	                    stringFeatures.push(feature);
	                }
	            }

	            for (let node of this.nodes) {
	                pointFeatures = pointFeatures.concat(node.toFeatureArray());
	            }

	            return polygonFeatures.concat(stringFeatures).concat(pointFeatures);
	        }
	    }

	    return {Node, Way, Relation}; 
	})();

	var xmlparser = (() => {
	    
	    function conditioned(evt) {
	        return evt.match(/^(.+?)\[(.+?)\]>$/g) != null;
	    }

	    function parseEvent(evt) {
	        let match = /^(.+?)\[(.+?)\]>$/g.exec(evt);
	        if (match) {
	            return {evt: match[1] + '>', exp: match[2]};
	        }
	        return {evt: evt};
	    }

	    function genConditionFunc(cond) {
	        let body = 'return ' + cond.replace(/(\$.+?)(?=[=!.])/g, 'node.$&') + ';';
	        return new Function('node', body);
	    }

	    return class {
	        constructor(opts) {
	            if (opts) {
	                this.queryParent = opts.queryParent? true : false;
	                this.progressive = opts.progressive;
	                if (this.queryParent) {
	                    this.parentMap = new WeakMap();
	                }
	            }
	            this.evtListeners = {};
	        }

	        parse(xml, parent, dir) {
	            dir = dir? dir + '.' : '';
	            let nodeRegEx = /<([^ >\/]+)(.*?)>/mg, nodeMatch = null, nodes = [];
	            while (nodeMatch = nodeRegEx.exec(xml)) {
	                let tag = nodeMatch[1], node = {$tag: tag}, fullTag = dir + tag; 

	                let attrText = nodeMatch[2].trim(), closed = false;
	                if (attrText.endsWith('/') || tag.startsWith('?') || tag.startsWith('!')) {
	                    closed = true;
	                }

	                let attRegEx1 = /([^ ]+?)="(.+?)"/g, attRegEx2 = /([^ ]+?)='(.+?)'/g;
	                let attMatch = null, hasAttrs = false;
	                while (attMatch = attRegEx1.exec(attrText)) {
	                    hasAttrs = true;
	                    node[attMatch[1]] = attMatch[2];
	                }
	                if (!hasAttrs)
	                    while (attMatch = attRegEx2.exec(attrText)) {
	                        hasAttrs = true;
	                        node[attMatch[1]] = attMatch[2];
	                    }

	                if (!hasAttrs && attrText !== '') {
	                    node.text = attrText;
	                }
	                if (this.progressive) {
	                    this.emit(`<${fullTag}>`, node, parent);
	                }

	                if (!closed) {
	                    let innerRegEx = new RegExp(`([^]+?)<\/${tag}>`, 'g');
	                    innerRegEx.lastIndex = nodeRegEx.lastIndex;
	                    let innerMatch = innerRegEx.exec(xml);
	                    if (innerMatch && innerMatch[1]) {
	                        nodeRegEx.lastIndex = innerRegEx.lastIndex;
	                        let innerNodes = this.parse(innerMatch[1], node, fullTag);
	                        if (innerNodes.length > 0) {
	                            node.$innerNodes = innerNodes;
	                        } else {
	                            node.$innerText = innerMatch[1];
	                        }
	                    }
	                }
	                if (this.queryParent && parent) {
	                    this.parentMap.set(node, parent);
	                }

	                if (this.progressive) {
	                    this.emit(`</${fullTag}>`, node, parent);
	                }

	                nodes.push(node);
	            }

	            return nodes;
	        }

	        getParent(node) {
	            if (this.queryParent) {
	                return this.parentMap.get(node);
	            }
	            return null;
	        }

	        $addListener(evt, func) {
	            let funcs = this.evtListeners[evt];
	            if (funcs) {
	                funcs.push(func);
	            } else {
	                this.evtListeners[evt] = [func];
	            }
	        }

	        // support javascript condition for the last tag
	        addListener(evt, func) {
	            if (conditioned(evt)) {
	                // func.prototype = evt;
	                evt = parseEvent(evt);    
	                func.condition = genConditionFunc(evt.exp);
	                evt = evt.evt;
	            }
	            this.$addListener(evt, func);
	        }

	        $removeListener(evt, func) {
	            let funcs = this.evtListeners[evt];
	            let idx = null;
	            if (funcs && (idx = funcs.indexOf(func)) >= 0) {
	                funcs.splice(idx, 1);
	            }
	        }

	        removeListener(evt, func) {
	            if (conditioned(evt)) {
	                evt = parseEvent(evt);    
	                evt = evt.evt;
	            }
	            this.$removeListener(evt, func);
	        }

	        emit(evt, ...args) {
	            let funcs = this.evtListeners[evt];
	            if (funcs) {
	                for (let func of funcs) {
	                    if (func.condition) {
	                        if (func.condition.apply(null, args) === true) {
	                            func.apply(null, args);
	                        }
	                    } else {
	                        func.apply(null, args);
	                    }
	                }
	            }
	        }

	        on(evt, func) {
	            this.addListener(evt, func);
	        }

	        off(evt, func) {
	            this.removeListener(evt, func);
	        }
	    };
	})();

	const {Node, Way, Relation} = osmobjs,
	    {purgeProps, RefElements} = utils,
	    XmlParser = xmlparser;

	var lib = (osm, opts) => {
	    let completeFeature = false, renderTagged = false, excludeWay = true;

	    const parseOpts = opts => {
	        if (opts) {
	            completeFeature = opts.completeFeature || opts.allFeatures? true : false;
	            renderTagged = opts.renderTagged? true : false;
	            let wayOpt = opts.suppressWay || opts.excludeWay;
	            if (wayOpt !== undefined && !wayOpt) {
	                excludeWay = false;
	            }
	        }
	    };

	    parseOpts(opts);

	    const detectFormat = osm => {
	        if (osm.elements) {
	            return 'json';
	        }
	        if (osm.indexOf('<osm') >= 0) {
	            return 'xml';
	        }
	        if (osm.trim().startsWith('{')) {
	            return 'json-raw';
	        }
	        return 'invalid';
	    };

	    let format = detectFormat(osm);

	    let refElements = new RefElements(), featureArray = [];

	    const analyzeFeaturesFromJson = osm => {
	        for (let elem of osm.elements) {
	            switch(elem.type) {
	                case 'node':
	                    let node = new Node(elem.id, refElements);
	                    if (elem.tags) {
	                        node.addTags(elem.tags);
	                    }
	                    node.addProps(purgeProps(elem, ['id', 'type', 'tags', 'lat', 'lon']));
	                    node.setLatLng(elem);
	                    break;

	                case 'way':
	                    let way = new Way(elem.id, refElements);
	                    if (elem.tags) {
	                        way.addTags(elem.tags);
	                    }
	                    way.addProps(purgeProps(elem, ['id', 'type', 'tags', 'nodes', 'geometry']));
	                    if (elem.nodes) {
	                        for (let n of elem.nodes) {
	                            way.addNodeRef(n);
	                        }
	                    } else if (elem.geometry) {
	                        way.setLatLngArray(elem.geometry);
	                    }
	                    break;

	                case 'relation':
	                    let relation = new Relation(elem.id, refElements);
	                      if (elem.bounds) {
	                        relation.setBounds([parseFloat(elem.bounds.minlon), parseFloat(elem.bounds.minlat), parseFloat(elem.bounds.maxlon), parseFloat(elem.bounds.maxlat)]);
	                    }
	                    if (elem.tags) {
	                        relation.addTags(elem.tags);
	                    }
	                    relation.addProps(purgeProps(elem, ['id', 'type', 'tags', 'bounds', 'members']));
	                    if (elem.members) {
	                        for (let member of elem.members) {
	                            relation.addMember(member);
	                        }
	                    }
	                    break;
	            }
	        }
	    };

	    const analyzeFeaturesFromXml = osm => {
	        const xmlParser = new XmlParser({progressive: true});

	        xmlParser.on('</osm.node>', node => {
	            let nd = new Node(node.id, refElements);
	            for (let [k, v] of Object.entries(node))
	                if (!k.startsWith('$') && ['id', 'lon', 'lat'].indexOf(k) < 0) {
	                    nd.addProp(k, v);
	                }
	            nd.setLatLng(node);
	            if (node.$innerNodes) {
	                for (let ind of node.$innerNodes) {
	                    if(ind.$tag === 'tag') {
	                        nd.addTag(ind.k, ind.v);
	                    }
	                }
	            }
	        });

	        xmlParser.on('</osm.way>', node => {
	            let way = new Way(node.id, refElements);
	            for (let [k, v] of Object.entries(node)) {
	                if (!k.startsWith('$') && ['id'].indexOf(k) < 0) {
	                    way.addProp(k, v);
	                }
	            }
	            if (node.$innerNodes) {
	                for (let ind of node.$innerNodes) {
	                    if (ind.$tag === 'nd') {
	                        if (ind.lon && ind.lat) {
	                            way.addLatLng(ind);
	                        } else if (ind.ref) {
	                            way.addNodeRef(ind.ref);
	                        }
	                    } else if (ind.$tag === 'tag')
	                        way.addTag(ind.k, ind.v);
	                }
	            }
	        });

	        xmlParser.on('<osm.relation>', node => {
	            new Relation(node.id, refElements);
	        });

	        xmlParser.on('</osm.relation.member>', (node, parent) => {
	            let relation = refElements.get(`relation/${parent.id}`);
	            let member = {
	                type: node.type,
	                role: node.role? node.role : '',
	                ref: node.ref
	            };
	            if (node.lat && node.lon) {
	                member.lat = node.lat, member.lon = node.lon, member.tags = {};
	                for (let [k, v] of Object.entries(node)) {
	                    if (!k.startsWith('$') && ['type', 'lat', 'lon'].indexOf(k) < 0) {
	                        member[k] = v;
	                    }
	                }
	            }
	            if (node.$innerNodes) {
	                let geometry = [];
	                let nodes = [];
	                for (let ind of node.$innerNodes) {
	                    if (ind.lat && ind.lon) {
	                        geometry.push(ind);
	                    } else {
	                        nodes.push(ind.ref);
	                    }
	                }
	                if (geometry.length > 0) {
	                    member.geometry = geometry;
	                } else if (nodes.length > 0) {
	                    member.nodes = nodes;
	                }
	            }
	            relation.addMember(member);
	        });

	        xmlParser.on('</osm.relation.bounds>', (node, parent) => {
	            refElements.get(`relation/${parent.id}`).setBounds([parseFloat(node.minlon), parseFloat(node.minlat), parseFloat(node.maxlon), parseFloat(node.maxlat)]);
	        });

	        xmlParser.on('</osm.relation.tag>', (node, parent) => {
	            refElements.get(`relation/${parent.id}`).addTag(node.k, node.v);
	        });
	        
	        xmlParser.parse(osm);
	    };

	    if (format === 'json-raw') {
	        osm = JSON.parse(osm);
	        if (osm.elements) {
	            format = 'json';
	        } else {
	            format = 'invalid';
	        }
	    }

	    if (format === 'json') {
	        analyzeFeaturesFromJson(osm);
	    } else if (format === 'xml') {
	        analyzeFeaturesFromXml(osm);
	    }

	    refElements.bindAll();

	    for (let v of refElements.values()) {
	        if (v.refCount <= 0 || (v.hasTag && renderTagged && !(v instanceof Way && excludeWay))) {
	            let features = v.toFeatureArray();
	            // return the first geometry of the first relation element
	            if (v instanceof Relation && !completeFeature && features.length > 0) {
	                return features[0].geometry;
	            }
	            featureArray = featureArray.concat(features);
	        }
	    }

	    return {type: 'FeatureCollection', features: featureArray};
	};

	var osm2geojson = /*@__PURE__*/getDefaultExportFromCjs(lib);

	class Converter {
	    constructor(format, data) {
	        /**
	         * Creates a blank GeoJSON feature collection.
	         * @returns A new GeoJSON feature collection with no features.
	         */
	        this.blankGeoJSON = () => ({
	            type: 'FeatureCollection',
	            features: [],
	        });
	        this._rawData = data;
	        this._format = format;
	        const converters = {
	            'topojson': this.loadTopoJson,
	            'osm': this.loadOsm,
	            'kml': this.loadXml,
	            'gpx': this.loadXml,
	            'tcx': this.loadXml,
	            'csv': this.loadCsv,
	            'tsv': this.loadCsv
	        };
	        this._conversionFn = converters[format];
	    }
	    async convert() {
	        if (!this._conversionFn) {
	            return new Promise((_, rej) => rej(`No converter exists for ${this._format}`));
	        }
	        else {
	            return this._conversionFn();
	        }
	    }
	    /**
	     * Load the XML data as GeoJSON
	     * @returns A promise resolving to a GeoJSON FeatureCollection
	     */
	    async loadXml() {
	        // Use the appropriate parser based on the format
	        const geojson = toGeoJson[this._format](new DOMParser().parseFromString(this._rawData, "text/xml"));
	        return geojson;
	    }
	    /**
	     * Loads and parses CSV data into a GeoJSON FeatureCollection.
	     * @returns A Promise that resolves with the GeoJSON FeatureCollection.
	     */
	    async loadCsv() {
	        // Define options for the csv2geojson library
	        let options = {}; // TODO allow CSV options
	        if (this._format === 'tsv') {
	            options.delimiter = '\t';
	        }
	        // Use the csv2geojson library to convert the CSV to GeoJSON
	        const geojson = await new Promise((resolve, reject) => {
	            csv2geojson_1.csv2geojson(this._rawData, options, (err, data) => {
	                if (err) {
	                    reject(err);
	                }
	                else {
	                    resolve(data);
	                }
	            });
	        });
	        return geojson;
	    }
	    /**
	     * Loads TopoJSON data and converts it into a GeoJSON FeatureCollection
	     */
	    async loadTopoJson() {
	        let topoJsonData = {};
	        try {
	            topoJsonData = JSON.parse(this._rawData);
	        }
	        catch (e) {
	            throw "Invalid TopoJson";
	        }
	        // Convert the data
	        let result = this.blankGeoJSON();
	        if (topoJsonData.type === "Topology" && topoJsonData.objects !== undefined) {
	            result = {
	                type: "FeatureCollection",
	                features: result.features = Object.keys(topoJsonData.objects).map(key => topojsonFeature(topoJsonData, key)).reduce((a, v) => [...a, ...v.features], [])
	            };
	        }
	        return result;
	    }
	    ;
	    /**
	     * Loads OSM data and converts it into a GeoJSON FeatureCollection
	     */
	    async loadOsm() {
	        return osm2geojson(this._rawData);
	    }
	}

	const libraries = {
	    'Converter': Converter
	};
	let subClass;
	self.addEventListener('message', e => {
	    const data = (e.data || e);
	    const post = (id, err, res, type) => {
	        postMessage({
	            type: type ? type : (err ? 'error' : 'response'),
	            id: id,
	            message: res,
	            error: err
	        });
	    };
	    const commands = {
	        'init': (msg) => {
	            const { id, command, message } = msg;
	            subClass = new libraries[command](message[0], message[1]);
	            // return the class' methods
	            const fns = [
	                ...Object.getOwnPropertyNames(libraries[command].prototype),
	                ...Object.keys(subClass)
	            ].map(key => [key, typeof libraries[command].prototype[key]])
	                .reduce((a, c) => ({ ...a, ...{ [c[0]]: c[1] } }), {});
	            post(id, undefined, fns, 'init_response');
	        },
	        'get': function (msg) {
	            const { id, command } = msg;
	            if (subClass && subClass[command]) {
	                post(id, undefined, subClass[command]);
	            }
	            else {
	                post(id, undefined, undefined);
	            }
	        },
	        'exec': function (msg) {
	            const { id, command, message } = msg;
	            if (subClass && subClass[command] && typeof subClass[command] === 'function') {
	                const cmd = subClass[command]
	                    .apply(subClass, message);
	                if (!!cmd && typeof cmd.then === 'function') {
	                    // It's a promise, so wait for it
	                    cmd
	                        .then(res => post(id, undefined, res))
	                        .catch(e => post(id, e));
	                }
	                else {
	                    // Not a promise, just return it
	                    post(id, undefined, cmd);
	                }
	            }
	            else {
	                // Error
	                post(id, new Error(`command "${command}" not found`));
	            }
	        }
	    };
	    if (commands[data.type]) {
	        commands[data.type](data);
	    }
	});

})();

",cC=null,oC=!1,lC?IC(bC,cC,oC):function(g,C,I){var A;return function(l){return A=A||AC(g,C,I),new Worker(A,l)}}(bC,cC,oC));const dC=()=>Math.random().toString(36).substring(2);class GC{constructor(g,C){this.initId=dC()+"-"+g,this.worker=new ZC,this.handlers=new Map,this.worker.onmessage=C=>{const I=C.data,A=this.handlers.get(I.id),l=this;if(A){if("response"===I.type&&A.resolve(I.message),"error"===I.type){const C=I.error||new Error(`Unknown error with ${g}`);A.reject(C)}"init_response"===I.type&&(this._=Object.keys(I.message).map((g=>{const C="function"==typeof I.message[g];return[g,function(){return C?l.exec(g)(...arguments):l.get(g)}]})).reduce(((g,C)=>({...g,[C[0]]:C[1]})),{}),A.resolve(this._))}},this.worker.postMessage({type:"init",id:this.initId,command:g,message:C})}onLoad(){return new Promise((g=>{void 0===this._?this.handlers.set(this.initId,{resolve:g,reject:g}):g(this._)}))}exec(g){const C=this;return function(...I){return new Promise(((A,l)=>{const b=dC()+"-"+g;C.handlers.set(b,{resolve:A,reject:l}),C.worker.postMessage({type:"exec",id:b,command:g,message:[...I]})}))}}get(g){return new Promise(((C,I)=>{const A=dC()+"-"+g;this.handlers.set(A,{resolve:C,reject:I}),this.worker.postMessage({type:"get",id:A,command:g,message:[]})}))}}const sC="test://http://example.com"!==new URL("test://http://example.com").href;async function eC(g,C,I){const A=await fetch(g,I?{signal:I.signal}:void 0);if(200==A.status){const g=await A.text();let I,l;return["kml","tcx","gpx"].indexOf(C)>=0||!(()=>{let g=!1;try{g="function"==typeof window.Worker}catch(C){g=!1}return g})()?(I=new $g(C,g),l=I.convert()):(I=new GC("Converter",[C,g]),l=I.exec("convert")()),await l}throw new Error(`Data fetch error: ${A.statusText}`)}const nC=(g,C)=>{const I=new AbortController,A=g.url.split("://")[0],l=g.url.replace(new RegExp(`^${A}://`),""),b=sC?(g=>{const C=new RegExp("^(https?)(//)");return g.replace(C,"$1:$2")})(l):l;return b&&eC(b,A,I).then((g=>C(null,g))).catch((g=>C(g))),{cancel:()=>{I.abort()}}};g.VectorTextProtocol=nC,g.addProtocols=g=>{_g.forEach((C=>{g.addProtocol(C,nC)}))},Object.defineProperty(g,"__esModule",{value:!0})}));
//# sourceMappingURL=maplibre-gl-vector-text-protocol.min.js.map