(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o 65535) { this.current = new ElementGroup(this.vertexBuffer.index, this.elementBuffer && this.elementBuffer.index, this.secondElementBuffer && this.secondElementBuffer.index); this.groups.push(this.current); } }; function ElementGroup(vertexStartIndex, elementStartIndex, secondElementStartIndex) { // the offset into the vertex buffer of the first vertex in this group this.vertexStartIndex = vertexStartIndex; this.elementStartIndex = elementStartIndex; this.secondElementStartIndex = secondElementStartIndex; this.elementLength = 0; this.vertexLength = 0; this.secondElementLength = 0; } },{}],12:[function(require,module,exports){ 'use strict'; var rbush = require('rbush'), Point = require('point-geometry'); module.exports = FeatureTree; function FeatureTree(getGeometry, getType) { this.getGeometry = getGeometry; this.getType = getType; this.rtree = rbush(9); this.toBeInserted = []; } FeatureTree.prototype.insert = function(bbox, bucket_name, feature) { bbox.bucket = bucket_name; bbox.feature = feature; this.toBeInserted.push(bbox); }; // bulk insert into tree FeatureTree.prototype._load = function() { this.rtree.load(this.toBeInserted); this.toBeInserted = []; }; // Finds features in this tile at a particular position. FeatureTree.prototype.query = function(args, callback) { if (this.toBeInserted.length) this._load(); var radius = args.params && args.params.radius || 0; radius *= 4096 / args.scale; var x = args.x, y = args.y; var matching = this.rtree.search([ x - radius, y - radius, x + radius, y + radius ]); if (args.params.buckets) { this.queryBuckets(matching, x, y, radius, args.params, callback); } else { this.queryFeatures(matching, x, y, radius, args.params, callback); } }; FeatureTree.prototype.queryFeatures = function(matching, x, y, radius, params, callback) { var result = []; for (var i = 0; i < matching.length; i++) { var feature = matching[i].feature; var type = this.getType(feature); var geometry = this.getGeometry(feature); if (params.bucket && matching[i].bucket !== params.bucket) continue; if (params.type && type !== params.type) continue; if (geometryContainsPoint(geometry, type, new Point(x, y), radius)) { var props = { _bucket: matching[i].bucket, _type: type }; if (params.geometry) { props._geometry = geometry; } for (var key in feature) { if (feature.hasOwnProperty(key) && key[0] !== '_') { props[key] = feature[key]; } } result.push(props); } } callback(null, result); }; // Lists all buckets that at the position. FeatureTree.prototype.queryBuckets = function(matching, x, y, radius, params, callback) { var buckets = []; for (var i = 0; i < matching.length; i++) { if (buckets.indexOf(matching[i].bucket) >= 0) continue; var feature = matching[i].feature; var type = this.getType(feature); var geometry = this.getGeometry(feature); if (geometryContainsPoint(geometry, type, new Point(x, y), radius)) { buckets.push(matching[i].bucket); } } callback(null, buckets); }; function geometryContainsPoint(rings, type, p, radius) { if (type === 'Point') { return pointContainsPoint(rings, p, radius); } else if (type === 'LineString') { return lineContainsPoint(rings, p, radius); } else if (type === 'Polygon') { return polyContainsPoint(rings, p) ? true : lineContainsPoint(rings, p, radius); } else { return false; } } // Code from http://stackoverflow.com/a/1501725/331379. function distToSegmentSquared(p, v, w) { var l2 = v.distSqr(w); if (l2 === 0) return p.distSqr(v); var t = ((p.x - v.x) * (w.x - v.x) + (p.y - v.y) * (w.y - v.y)) / l2; if (t < 0) return p.distSqr(v); if (t > 1) return p.distSqr(w); return p.distSqr(w.sub(v)._mult(t)._add(v)); } function lineContainsPoint(rings, p, radius) { var r = radius * radius; for (var i = 0; i < rings.length; i++) { var ring = rings[i]; for (var j = 1; j < ring.length; j++) { // Find line segments that have a distance <= radius^2 to p // In that case, we treat the line as "containing point p". var v = ring[j-1], w = ring[j]; if (distToSegmentSquared(p, v, w) < r) return true; } } return false; } // point in polygon ray casting algorithm function polyContainsPoint(rings, p) { var c = false, ring, p1, p2; for (var k = 0; k < rings.length; k++) { ring = rings[k]; for (var i = 0, j = ring.length - 1; i < ring.length; j = i++) { p1 = ring[i]; p2 = ring[j]; if (((p1.y > p.y) != (p2.y > p.y)) && (p.x < (p2.x - p1.x) * (p.y - p1.y) / (p2.y - p1.y) + p1.x)) { c = !c; } } } return c; } function pointContainsPoint(rings, p, radius) { var r = radius * radius; for (var i = 0; i < rings.length; i++) { var ring = rings[i]; for (var j = 0; j < ring.length; j++) { if (ring[j].distSqr(p) <= r) return true; } } return false; } },{"point-geometry":98,"rbush":100}],13:[function(require,module,exports){ 'use strict'; var ElementGroups = require('./elementgroups.js'); module.exports = FillBucket; function FillBucket(info, buffers, placement, elementGroups) { this.info = info; this.buffers = buffers; this.elementGroups = elementGroups || new ElementGroups(buffers.fillVertex, buffers.fillElement, buffers.outlineElement); } FillBucket.prototype.addFeatures = function() { var features = this.features; for (var i = 0; i < features.length; i++) { var feature = features[i]; this.addFeature(feature.loadGeometry()); } }; FillBucket.prototype.addFeature = function(lines) { for (var i = 0; i < lines.length; i++) { this.addFill(lines[i]); } }; FillBucket.prototype.addFill = function(vertices) { if (vertices.length < 3) { //console.warn('a fill must have at least three vertices'); return; } // Calculate the total number of vertices we're going to produce so that we // can resize the buffer beforehand, or detect whether the current line // won't fit into the buffer anymore. // In order to be able to use the vertex buffer for drawing the antialiased // outlines, we separate all polygon vertices with a degenerate (out-of- // viewplane) vertex. var len = vertices.length; // Check whether this geometry buffer can hold all the required vertices. this.elementGroups.makeRoomFor(len + 1); var elementGroup = this.elementGroups.current; var fillVertex = this.buffers.fillVertex; var fillElement = this.buffers.fillElement; var outlineElement = this.buffers.outlineElement; // Start all lines with a degenerate vertex elementGroup.vertexLength++; // We're generating triangle fans, so we always start with the first coordinate in this polygon. var firstIndex = fillVertex.index - elementGroup.vertexStartIndex, prevIndex, currentIndex, currentVertex; for (var i = 0; i < vertices.length; i++) { currentIndex = fillVertex.index - elementGroup.vertexStartIndex; currentVertex = vertices[i]; fillVertex.add(currentVertex.x, currentVertex.y); elementGroup.vertexLength++; // Only add triangles that have distinct vertices. if (i >= 2 && (currentVertex.x !== vertices[0].x || currentVertex.y !== vertices[0].y)) { fillElement.add(firstIndex, prevIndex, currentIndex); elementGroup.elementLength++; } if (i >= 1) { outlineElement.add(prevIndex, currentIndex); elementGroup.secondElementLength++; } prevIndex = currentIndex; } }; FillBucket.prototype.hasData = function() { return !!this.elementGroups.current; }; },{"./elementgroups.js":11}],14:[function(require,module,exports){ 'use strict'; var ElementGroups = require('./elementgroups.js'); module.exports = LineBucket; function LineBucket(info, buffers, placement, elementGroups) { this.info = info; this.buffers = buffers; this.elementGroups = elementGroups || new ElementGroups(buffers.lineVertex, buffers.lineElement); } LineBucket.prototype.addFeatures = function() { var features = this.features; for (var i = 0; i < features.length; i++) { var feature = features[i]; this.addFeature(feature.loadGeometry()); } }; LineBucket.prototype.addFeature = function(lines) { var info = this.info; for (var i = 0; i < lines.length; i++) { this.addLine(lines[i], info['line-join'], info['line-cap'], info['line-miter-limit'], info['line-round-limit']); } }; LineBucket.prototype.addLine = function(vertices, join, cap, miterLimit, roundLimit) { if (vertices.length < 2) { //console.warn('a line must have at least two vertices'); return; } if (join === 'bevel') miterLimit = 1.05; var len = vertices.length, firstVertex = vertices[0], lastVertex = vertices[len - 1], closed = firstVertex.equals(lastVertex); var lineVertex = this.buffers.lineVertex; var lineElement = this.buffers.lineElement; // we could be more precies, but it would only save a negligible amount of space this.elementGroups.makeRoomFor(len * 4); var elementGroup = this.elementGroups.current; var vertexStartIndex = elementGroup.vertexStartIndex; if (len == 2 && closed) { // console.warn('a line may not have coincident points'); return; } var beginCap = cap, endCap = closed ? 'butt' : cap, flip = 1, distance = 0, currentVertex, prevVertex, nextVertex, prevNormal, nextNormal; // the last three vertices added var e1, e2, e3; if (closed) { currentVertex = vertices[len - 2]; nextNormal = firstVertex.sub(currentVertex)._unit()._perp(); } for (var i = 0; i < len; i++) { nextVertex = closed && i === len - 1 ? vertices[1] : // if the line is closed, we treat the last vertex like the first vertices[i + 1]; // just the next vertex // if two consecutive vertices exist, skip the current one if (nextVertex && vertices[i].equals(nextVertex)) continue; if (nextNormal) prevNormal = nextNormal; if (currentVertex) prevVertex = currentVertex; currentVertex = vertices[i]; // Calculate how far along the line the currentVertex is if (prevVertex) distance += currentVertex.dist(prevVertex); // Calculate the normal towards the next vertex in this line. In case // there is no next vertex, pretend that the line is continuing straight, // meaning that we are just using the previous normal. nextNormal = nextVertex ? nextVertex.sub(currentVertex)._unit()._perp() : prevNormal; // If we still don't have a previous normal, this is the beginning of a // non-closed line, so we're doing a straight "join". prevNormal = prevNormal || nextNormal; // Determine the normal of the join extrusion. It is the angle bisector // of the segments between the previous line and the next line. var joinNormal = prevNormal.add(nextNormal)._unit(); /* joinNormal prevNormal * ↖ ↑ * .________. prevVertex * | * nextNormal ← | currentVertex * | * nextVertex ! * */ // Calculate the length of the miter (the ratio of the miter to the width). // Find the cosine of the angle between the next and join normals // using dot product. The inverse of that is the miter length. var cosHalfAngle = joinNormal.x * nextNormal.x + joinNormal.y * nextNormal.y; var miterLength = 1 / cosHalfAngle; // Whether any vertices have been var startOfLine = e1 === undefined || e2 === undefined; // The join if a middle vertex, otherwise the cap. var middleVertex = prevVertex && nextVertex; var currentJoin = middleVertex ? join : nextVertex ? beginCap : endCap; if (middleVertex && currentJoin === 'round' && miterLength < roundLimit) { currentJoin = 'miter'; } if (currentJoin === 'miter' && miterLength > miterLimit) { currentJoin = 'bevel'; } if (currentJoin === 'bevel') { // The maximum extrude length is 63 / 256 = 4 times the width of the line // so if miterLength >= 4 we need to draw a different type of bevel where. if (miterLength > 4) currentJoin = 'flipbevel'; // If the miterLength is really small and the line bevel wouldn't be visible, // just draw a miter join to save a triangle. if (miterLength < miterLimit) currentJoin = 'miter'; } // Mitered joins if (currentJoin === 'miter') { // scale the unit vector by the miter length joinNormal._mult(miterLength); addCurrentVertex(joinNormal, 0, 0, false); } else if (currentJoin === 'flipbevel') { // miter is too big, flip the direction to make a beveled join if (miterLength > 100) { // Almost parallel lines flip = -flip; joinNormal = nextNormal; } else { var bevelLength = miterLength * prevNormal.add(nextNormal).mag() / prevNormal.sub(nextNormal).mag(); joinNormal._perp()._mult(flip * bevelLength); flip = -flip; } addCurrentVertex(joinNormal, 0, 0, false); // All other types of joins } else { var offsetA, offsetB; if (currentJoin === 'bevel') { var dir = prevNormal.x * nextNormal.y - prevNormal.y * nextNormal.x; var offset = -Math.sqrt(miterLength * miterLength - 1); if (flip * dir > 0) { offsetB = 0; offsetA = offset; } else { offsetA = 0; offsetB = offset; } } else if (currentJoin === 'square') { offsetA = offsetB = 1; } else { offsetA = offsetB = 0; } // Close previous segment with a butt or a square cap or bevel if (!startOfLine) { addCurrentVertex(prevNormal, offsetA, offsetB, false); } // Add round cap or linejoin at end of segment if (!startOfLine && currentJoin === 'round') { addCurrentVertex(prevNormal, 1, 1, true); } // Segment include cap are done, unset vertices to disconnect segments. // Or leave them to create a bevel. if (startOfLine || currentJoin !== 'bevel') { e1 = e2 = -1; flip = 1; } // Add round cap before first segment if (startOfLine && beginCap === 'round') { addCurrentVertex(nextNormal, -1, -1, true); } // Start next segment with a butt or square cap or bevel if (nextVertex) { addCurrentVertex(nextNormal, -offsetA, -offsetB, false); } } } /* * Adds two vertices to the buffer that are * normal and -normal from the currentVertex. * * endBox moves the extrude one unit in the direction of the line * to create square or round cap. * * endLeft and endRight shifts the extrude along the line * endLeft === 1 moves the extrude in the direction of the line * endLeft === -1 moves the extrude in the reverse direction */ function addCurrentVertex(normal, endLeft, endRight, round) { var tx = round ? 1 : 0; var extrude; extrude = normal.mult(flip); if (endLeft) extrude._sub(normal.perp()._mult(endLeft)); e3 = lineVertex.add(currentVertex, extrude, tx, 0, distance) - vertexStartIndex; if (e1 >= 0 && e2 >= 0) { lineElement.add(e1, e2, e3); elementGroup.elementLength++; } e1 = e2; e2 = e3; extrude = normal.mult(-flip); if (endRight) extrude._sub(normal.perp()._mult(endRight)); e3 = lineVertex.add(currentVertex, extrude, tx, 1, distance) - vertexStartIndex; if (e1 >= 0 && e2 >= 0) { lineElement.add(e1, e2, e3); elementGroup.elementLength++; } e1 = e2; e2 = e3; elementGroup.vertexLength += 2; } }; LineBucket.prototype.hasData = function() { return !!this.elementGroups.current; }; },{"./elementgroups.js":11}],15:[function(require,module,exports){ 'use strict'; module.exports = RasterBucket; function RasterBucket(info) { this.info = info; } },{}],16:[function(require,module,exports){ 'use strict'; var ElementGroups = require('./elementgroups.js'); var Anchor = require('../symbol/anchor.js'); var interpolate = require('../symbol/interpolate.js'); var Point = require('point-geometry'); var resolveTokens = require('../util/token.js'); var Placement = require('../symbol/placement.js'); var Shaping = require('../symbol/shaping.js'); var resolveText = require('../symbol/resolvetext.js'); module.exports = SymbolBucket; var fullRange = [2 * Math.PI , 0]; function SymbolBucket(info, buffers, collision, elementGroups) { this.info = info; this.buffers = buffers; this.collision = collision; if (info['symbol-placement'] === 'line') { if (!info.hasOwnProperty('text-rotation-alignment')) { info['text-rotation-alignment'] = 'map'; } if (!info.hasOwnProperty('icon-rotation-alignment')) { info['icon-rotation-alignment'] = 'map'; } info['symbol-avoid-edges'] = true; } if (elementGroups) { this.elementGroups = elementGroups; } else { this.elementGroups = { text: new ElementGroups(buffers.glyphVertex), icon: new ElementGroups(buffers.iconVertex) }; } } SymbolBucket.prototype.addFeatures = function() { var info = this.info; var features = this.features; var textFeatures = this.textFeatures; var horizontalAlign = 0.5; if (info['text-horizontal-align'] === 'right') horizontalAlign = 1; else if (info['text-horizontal-align'] === 'left') horizontalAlign = 0; var verticalAlign = 0.5; if (info['text-vertical-align'] === 'bottom') verticalAlign = 1; else if (info['text-vertical-align'] === 'top') verticalAlign = 0; var justify = 0.5; if (info['text-justify'] === 'right') justify = 1; else if (info['text-justify'] === 'left') justify = 0; var oneEm = 24; var lineHeight = info['text-line-height'] * oneEm; var maxWidth = info['symbol-placement'] !== 'line' && info['text-max-width'] * oneEm; var spacing = info['text-letter-spacing'] * oneEm; var fontstack = info['text-font']; var textOffset = [info['text-offset'][0] * oneEm, info['text-offset'][1] * oneEm]; for (var k = 0; k < features.length; k++) { var feature = features[k]; var text = textFeatures[k]; var lines = feature.loadGeometry(); var shaping = false; if (text) { shaping = Shaping.shape(text, fontstack, this.stacks, maxWidth, lineHeight, horizontalAlign, verticalAlign, justify, spacing, textOffset); } var image = false; if (this.sprite && this.info['icon-image']) { image = this.sprite[resolveTokens(feature.properties, info['icon-image'])]; if (image) { // match glyph tex object. TODO change image.w = image.width; image.h = image.height; if (image.sdf) this.elementGroups.sdfIcons = true; } } if (!shaping && !image) continue; this.addFeature(lines, this.stacks, shaping, image); } }; function byScale(a, b) { return a.scale - b.scale; } SymbolBucket.prototype.addFeature = function(lines, faces, shaping, image) { var info = this.info; var collision = this.collision; var minScale = 0.5; var glyphSize = 24; var horizontalText = info['text-rotation-alignment'] === 'viewport', horizontalIcon = info['icon-rotation-alignment'] === 'viewport', fontScale = info['text-max-size'] / glyphSize, textBoxScale = collision.tilePixelRatio * fontScale, iconBoxScale = collision.tilePixelRatio * info['icon-max-size'], iconWithoutText = info['text-optional'] || !shaping, textWithoutIcon = info['icon-optional'] || !image, avoidEdges = info['symbol-avoid-edges']; for (var i = 0; i < lines.length; i++) { var line = lines[i]; var anchors; if (info['symbol-placement'] === 'line') { // Line labels anchors = interpolate(line, info['symbol-min-distance'], minScale, collision.maxPlacementScale, collision.tilePixelRatio); // Sort anchors by segment so that we can start placement with the // anchors that can be shown at the lowest zoom levels. anchors.sort(byScale); } else { // Point labels anchors = [new Anchor(line[0].x, line[0].y, 0, minScale)]; } // TODO: figure out correct ascender height. var origin = new Point(0, -17); for (var j = 0, len = anchors.length; j < len; j++) { var anchor = anchors[j]; var inside = !(anchor.x < 0 || anchor.x > 4096 || anchor.y < 0 || anchor.y > 4096); if (avoidEdges && !inside) continue; // Calculate the scales at which the text and icons can be first shown without overlap var glyph; var icon; var glyphScale = null; var iconScale = null; if (shaping) { glyph = Placement.getGlyphs(anchor, origin, shaping, faces, textBoxScale, horizontalText, line, info); glyphScale = info['text-allow-overlap'] ? glyph.minScale : collision.getPlacementScale(glyph.boxes, glyph.minScale, avoidEdges); if (!glyphScale && !iconWithoutText) continue; } if (image) { icon = Placement.getIcon(anchor, image, iconBoxScale, line, info); iconScale = info['icon-allow-overlap'] ? icon.minScale : collision.getPlacementScale(icon.boxes, icon.minScale, avoidEdges); if (!iconScale && !textWithoutIcon) continue; } if (!iconWithoutText && !textWithoutIcon) { iconScale = glyphScale = Math.max(iconScale, glyphScale); } else if (!textWithoutIcon && glyphScale) { glyphScale = Math.max(iconScale, glyphScale); } else if (!iconWithoutText && iconScale) { iconScale = Math.max(iconScale, glyphScale); } // Get the rotation ranges it is safe to show the glyphs var glyphRange = (!glyphScale || info['text-allow-overlap']) ? fullRange : collision.getPlacementRange(glyph.boxes, glyphScale, horizontalText); var iconRange = (!iconScale || info['icon-allow-overlap']) ? fullRange : collision.getPlacementRange(icon.boxes, iconScale, horizontalIcon); var maxRange = [ Math.min(iconRange[0], glyphRange[0]), Math.max(iconRange[1], glyphRange[1])]; if (!iconWithoutText && !textWithoutIcon) { iconRange = glyphRange = maxRange; } else if (!textWithoutIcon) { glyphRange = maxRange; } else if (!iconWithoutText) { iconRange = maxRange; } // Insert final placement into collision tree and add glyphs/icons to buffers if (glyphScale) { if (!info['text-ignore-placement']) { collision.insert(glyph.boxes, anchor, glyphScale, glyphRange, horizontalText); } if (inside) this.addSymbols(this.buffers.glyphVertex, this.elementGroups.text, glyph.shapes, glyphScale, glyphRange); } if (iconScale) { if (!info['icon-ignore-placement']) { collision.insert(icon.boxes, anchor, iconScale, iconRange, horizontalIcon); } if (inside) this.addSymbols(this.buffers.iconVertex, this.elementGroups.icon, icon.shapes, iconScale, iconRange); } } } }; SymbolBucket.prototype.addSymbols = function(buffer, elementGroups, symbols, scale, placementRange) { var zoom = this.collision.zoom; elementGroups.makeRoomFor(0); var elementGroup = elementGroups.current; var placementZoom = Math.log(scale) / Math.LN2 + zoom; for (var k = 0; k < symbols.length; k++) { var symbol = symbols[k], tl = symbol.tl, tr = symbol.tr, bl = symbol.bl, br = symbol.br, tex = symbol.tex, angle = symbol.angle, anchor = symbol.anchor, minZoom = Math.max(zoom + Math.log(symbol.minScale) / Math.LN2, placementZoom), maxZoom = Math.min(zoom + Math.log(symbol.maxScale) / Math.LN2, 25); if (maxZoom <= minZoom) continue; // Lower min zoom so that while fading out the label it can be shown outside of collision-free zoom levels if (minZoom === placementZoom) minZoom = 0; // first triangle buffer.add(anchor.x, anchor.y, tl.x, tl.y, tex.x, tex.y, angle, minZoom, placementRange, maxZoom, placementZoom); buffer.add(anchor.x, anchor.y, tr.x, tr.y, tex.x + tex.w, tex.y, angle, minZoom, placementRange, maxZoom, placementZoom); buffer.add(anchor.x, anchor.y, bl.x, bl.y, tex.x, tex.y + tex.h, angle, minZoom, placementRange, maxZoom, placementZoom); // second triangle buffer.add(anchor.x, anchor.y, tr.x, tr.y, tex.x + tex.w, tex.y, angle, minZoom, placementRange, maxZoom, placementZoom); buffer.add(anchor.x, anchor.y, bl.x, bl.y, tex.x, tex.y + tex.h, angle, minZoom, placementRange, maxZoom, placementZoom); buffer.add(anchor.x, anchor.y, br.x, br.y, tex.x + tex.w, tex.y + tex.h, angle, minZoom, placementRange, maxZoom, placementZoom); elementGroup.vertexLength += 6; } }; SymbolBucket.prototype.getDependencies = function(tile, actor, callback) { var firstdone = false; var firsterr; this.getTextDependencies(tile, actor, done); this.getIconDependencies(tile, actor, done); function done(err) { if (err || firstdone) callback(err); firstdone = true; firsterr = err; } }; SymbolBucket.prototype.getIconDependencies = function(tile, actor, callback) { var bucket = this; if (this.info['icon-image']) { if (SymbolBucket.sprite) { this.sprite = SymbolBucket.sprite; callback(); } else { actor.send('get sprite json', {}, function(err, data) { SymbolBucket.sprite = bucket.sprite = data.sprite; callback(err); }); } } else { callback(); } }; SymbolBucket.prototype.getTextDependencies = function(tile, actor, callback) { var features = this.features; var info = this.info; if (tile.stacks === undefined) tile.stacks = {}; var stacks = this.stacks = tile.stacks; var fontstack = info['text-font']; if (stacks[fontstack] === undefined) { stacks[fontstack] = { glyphs: {}, rects: {} }; } var stack = stacks[fontstack]; var data = resolveText(features, info, stack.glyphs); this.textFeatures = data.textFeatures; actor.send('get glyphs', { id: tile.id, fontstack: fontstack, codepoints: data.codepoints }, function(err, newstack) { if (err) return callback(err); var newglyphs = newstack.glyphs; var newrects = newstack.rects; var glyphs = stack.glyphs; var rects = stack.rects; for (var codepoint in newglyphs) { glyphs[codepoint] = newglyphs[codepoint]; rects[codepoint] = newrects[codepoint]; } callback(); }); }; SymbolBucket.prototype.hasData = function() { return !!this.elementGroups.text.current || !!this.elementGroups.icon.current; }; },{"../symbol/anchor.js":57,"../symbol/interpolate.js":62,"../symbol/placement.js":63,"../symbol/resolvetext.js":64,"../symbol/shaping.js":66,"../util/token.js":85,"./elementgroups.js":11,"point-geometry":98}],17:[function(require,module,exports){ 'use strict'; module.exports = LatLng; function LatLng(lat, lng) { if (isNaN(lat) || isNaN(lng)) { throw new Error('Invalid LatLng object: (' + lat + ', ' + lng + ')'); } this.lat = +lat; this.lng = +lng; } // constructs LatLng from an array if necessary LatLng.convert = function (a) { if (a instanceof LatLng) { return a; } if (Array.isArray(a)) { return new LatLng(a[0], a[1]); } return a; }; },{}],18:[function(require,module,exports){ 'use strict'; module.exports = LatLngBounds; var LatLng = require('./latlng.js'); function LatLngBounds(sw, ne) { if (!sw) return; var latlngs = ne ? [sw, ne] : sw; for (var i = 0, len = latlngs.length; i < len; i++) { this.extend(latlngs[i]); } } LatLngBounds.prototype = { // extend the bounds to contain the given point or bounds extend: function (obj) { var sw = this._sw, ne = this._ne, sw2, ne2; if (obj instanceof LatLng) { sw2 = obj; ne2 = obj; } else if (obj instanceof LatLngBounds) { sw2 = obj._sw; ne2 = obj._ne; if (!sw2 || !ne2) return this; } else { return obj ? this.extend(LatLng.convert(obj) || LatLngBounds.convert(obj)) : this; } if (!sw && !ne) { this._sw = new LatLng(sw2.lat, sw2.lng); this._ne = new LatLng(ne2.lat, ne2.lng); } else { sw.lat = Math.min(sw2.lat, sw.lat); sw.lng = Math.min(sw2.lng, sw.lng); ne.lat = Math.max(ne2.lat, ne.lat); ne.lng = Math.max(ne2.lng, ne.lng); } return this; }, getCenter: function () { return new LatLng((this._sw.lat + this._ne.lat) / 2, (this._sw.lng + this._ne.lng) / 2); }, getSouthWest: function () { return this._sw; }, getNorthEast: function () { return this._ne; }, getNorthWest: function () { return new LatLng(this.getNorth(), this.getWest()); }, getSouthEast: function () { return new LatLng(this.getSouth(), this.getEast()); }, getWest: function () { return this._sw.lng; }, getSouth: function () { return this._sw.lat; }, getEast: function () { return this._ne.lng; }, getNorth: function () { return this._ne.lat; } }; // constructs LatLngBounds from an array if necessary LatLngBounds.convert = function (a) { if (!a || a instanceof LatLngBounds) return a; return new LatLngBounds(a); }; },{"./latlng.js":17}],19:[function(require,module,exports){ 'use strict'; var LatLng = require('./latlng.js'), Point = require('point-geometry'); module.exports = Transform; // A single transform, generally used for a single tile to be scaled, rotated, and zoomed. function Transform(minZoom, maxZoom) { this.tileSize = 512; // constant this._minZoom = minZoom || 0; this._maxZoom = maxZoom || 22; this.latRange = [-85, 85]; this.width = 0; this.height = 0; this.zoom = 0; this.center = new LatLng(0, 0); this.angle = 0; } Transform.prototype = { get minZoom() { return this._minZoom; }, set minZoom(zoom) { this._minZoom = zoom; this.zoom = Math.max(this.zoom, zoom); }, get maxZoom() { return this._maxZoom; }, set maxZoom(zoom) { this._maxZoom = zoom; this.zoom = Math.min(this.zoom, zoom); }, get worldSize() { return this.tileSize * this.scale; }, get centerPoint() { return this.size._div(2); }, get size() { return new Point(this.width, this.height); }, get bearing() { return -this.angle / Math.PI * 180; }, set bearing(bearing) { // confine the angle to within [-180,180] bearing = ((((bearing + 180) % 360) + 360) % 360) - 180; this.angle = -bearing * Math.PI / 180; }, get zoom() { return this._zoom; }, set zoom(zoom) { zoom = Math.min(Math.max(zoom, this.minZoom), this.maxZoom); this._zoom = zoom; this.scale = this.zoomScale(zoom); this.tileZoom = Math.floor(zoom); this.zoomFraction = zoom - this.tileZoom; this._constrain(); }, zoomScale: function(zoom) { return Math.pow(2, zoom); }, scaleZoom: function(scale) { return Math.log(scale) / Math.LN2; }, project: function(latlng, worldSize) { return new Point( this.lngX(latlng.lng, worldSize), this.latY(latlng.lat, worldSize)); }, unproject: function(point, worldSize) { return new LatLng( this.yLat(point.y, worldSize), this.xLng(point.x, worldSize)); }, get x() { return this.lngX(this.center.lng); }, get y() { return this.latY(this.center.lat); }, get point() { return new Point(this.x, this.y); }, // lat/lon <-> absolute pixel coords convertion lngX: function(lon, worldSize) { return (180 + lon) * (worldSize || this.worldSize) / 360; }, // latitude to absolute y coord latY: function(lat, worldSize) { var y = 180 / Math.PI * Math.log(Math.tan(Math.PI / 4 + lat * Math.PI / 360)); return (180 - y) * (worldSize || this.worldSize) / 360; }, xLng: function(x, worldSize) { return x * 360 / (worldSize || this.worldSize) - 180; }, yLat: function(y, worldSize) { var y2 = 180 - y * 360 / (worldSize || this.worldSize); return 360 / Math.PI * Math.atan(Math.exp(y2 * Math.PI / 180)) - 90; }, panBy: function(offset) { var point = this.centerPoint._add(offset); this.center = this.pointLocation(point); this._constrain(); }, setZoomAround: function(zoom, center) { var p = this.locationPoint(center), p1 = this.size._sub(p), latlng = this.pointLocation(p1); this.zoom = zoom; this.panBy(p1.sub(this.locationPoint(latlng))); }, setBearingAround: function(bearing, center) { var offset = this.locationPoint(center).sub(this.centerPoint); this.panBy(offset); this.bearing = bearing; this.panBy(offset.mult(-1)); }, locationPoint: function(latlng) { var p = this.project(latlng); return this.centerPoint._sub(this.point._sub(p)._rotate(this.angle)); }, pointLocation: function(p) { var p2 = this.centerPoint._sub(p)._rotate(-this.angle); return this.unproject(this.point.sub(p2)); }, locationCoordinate: function(latlng) { var k = this.zoomScale(this.tileZoom) / this.worldSize; return { column: this.lngX(latlng.lng) * k, row: this.latY(latlng.lat) * k, zoom: this.tileZoom }; }, pointCoordinate: function(tileCenter, p) { var zoomFactor = this.zoomScale(this.zoomFraction), kt = this.zoomScale(this.tileZoom - tileCenter.zoom), p2 = this.centerPoint._sub(p)._rotate(-this.angle)._div(this.tileSize * zoomFactor); return { column: tileCenter.column * kt - p2.x, row: tileCenter.row * kt - p2.y, zoom: this.tileZoom }; }, _constrain: function() { if (!this.center) return; var minY, maxY, minX, maxX, sy, sx, x2, y2, size = this.size; if (this.latRange) { minY = this.latY(this.latRange[1]); maxY = this.latY(this.latRange[0]); sy = maxY - minY < size.y ? size.y / (maxY - minY) : 0; } if (this.lngRange) { minX = this.lngX(this.lngRange[0]); maxX = this.lngX(this.lngRange[1]); sx = maxX - minX < size.x ? size.x / (maxX - minX) : 0; } // how much the map should scale to fit the screen into given latitude/longitude ranges var s = Math.max(sx || 0, sy || 0); if (s) { this.center = this.unproject(new Point( sx ? (maxX + minX) / 2 : this.x, sy ? (maxY + minY) / 2 : this.y)); this.zoom += this.scaleZoom(s); return; } if (this.latRange) { var y = this.y, h2 = size.y / 2; if (y - h2 < minY) y2 = minY + h2; if (y + h2 > maxY) y2 = maxY - h2; } if (this.lngRange) { var x = this.x, w2 = size.x / 2; if (x - w2 < minX) x2 = minX + w2; if (x + w2 > maxX) x2 = maxX - w2; } // pan the map if the screen goes off the range if (x2 !== undefined || y2 !== undefined) { this.center = this.unproject(new Point( x2 !== undefined ? x2 : this.x, y2 !== undefined ? y2 : this.y)); } } }; },{"./latlng.js":17,"point-geometry":98}],20:[function(require,module,exports){ // Font data From Hershey Simplex Font // http://paulbourke.net/dataformats/hershey/ var simplex_font = { " ": [16, []], "!": [10, [5, 21, 5, 7, -1, -1, 5, 2, 4, 1, 5, 0, 6, 1, 5, 2]], "\"": [16, [4, 21, 4, 14, -1, -1, 12, 21, 12, 14]], "#": [21, [11, 25, 4, -7, -1, -1, 17, 25, 10, -7, -1, -1, 4, 12, 18, 12, -1, -1, 3, 6, 17, 6]], "$": [20, [8, 25, 8, -4, -1, -1, 12, 25, 12, -4, -1, -1, 17, 18, 15, 20, 12, 21, 8, 21, 5, 20, 3, 18, 3, 16, 4, 14, 5, 13, 7, 12, 13, 10, 15, 9, 16, 8, 17, 6, 17, 3, 15, 1, 12, 0, 8, 0, 5, 1, 3, 3]], "%": [24, [21, 21, 3, 0, -1, -1, 8, 21, 10, 19, 10, 17, 9, 15, 7, 14, 5, 14, 3, 16, 3, 18, 4, 20, 6, 21, 8, 21, 10, 20, 13, 19, 16, 19, 19, 20, 21, 21, -1, -1, 17, 7, 15, 6, 14, 4, 14, 2, 16, 0, 18, 0, 20, 1, 21, 3, 21, 5, 19, 7, 17, 7]], "&": [26, [23, 12, 23, 13, 22, 14, 21, 14, 20, 13, 19, 11, 17, 6, 15, 3, 13, 1, 11, 0, 7, 0, 5, 1, 4, 2, 3, 4, 3, 6, 4, 8, 5, 9, 12, 13, 13, 14, 14, 16, 14, 18, 13, 20, 11, 21, 9, 20, 8, 18, 8, 16, 9, 13, 11, 10, 16, 3, 18, 1, 20, 0, 22, 0, 23, 1, 23, 2]], "'": [10, [5, 19, 4, 20, 5, 21, 6, 20, 6, 18, 5, 16, 4, 15]], "(": [14, [11, 25, 9, 23, 7, 20, 5, 16, 4, 11, 4, 7, 5, 2, 7, -2, 9, -5, 11, -7]], ")": [14, [3, 25, 5, 23, 7, 20, 9, 16, 10, 11, 10, 7, 9, 2, 7, -2, 5, -5, 3, -7]], "*": [16, [8, 21, 8, 9, -1, -1, 3, 18, 13, 12, -1, -1, 13, 18, 3, 12]], "+": [26, [13, 18, 13, 0, -1, -1, 4, 9, 22, 9]], ",": [10, [6, 1, 5, 0, 4, 1, 5, 2, 6, 1, 6, -1, 5, -3, 4, -4]], "-": [26, [4, 9, 22, 9]], ".": [10, [5, 2, 4, 1, 5, 0, 6, 1, 5, 2]], "/": [22, [20, 25, 2, -7]], "0": [20, [9, 21, 6, 20, 4, 17, 3, 12, 3, 9, 4, 4, 6, 1, 9, 0, 11, 0, 14, 1, 16, 4, 17, 9, 17, 12, 16, 17, 14, 20, 11, 21, 9, 21]], "1": [20, [6, 17, 8, 18, 11, 21, 11, 0]], "2": [20, [4, 16, 4, 17, 5, 19, 6, 20, 8, 21, 12, 21, 14, 20, 15, 19, 16, 17, 16, 15, 15, 13, 13, 10, 3, 0, 17, 0]], "3": [20, [5, 21, 16, 21, 10, 13, 13, 13, 15, 12, 16, 11, 17, 8, 17, 6, 16, 3, 14, 1, 11, 0, 8, 0, 5, 1, 4, 2, 3, 4]], "4": [20, [13, 21, 3, 7, 18, 7, -1, -1, 13, 21, 13, 0]], "5": [20, [15, 21, 5, 21, 4, 12, 5, 13, 8, 14, 11, 14, 14, 13, 16, 11, 17, 8, 17, 6, 16, 3, 14, 1, 11, 0, 8, 0, 5, 1, 4, 2, 3, 4]], "6": [20, [16, 18, 15, 20, 12, 21, 10, 21, 7, 20, 5, 17, 4, 12, 4, 7, 5, 3, 7, 1, 10, 0, 11, 0, 14, 1, 16, 3, 17, 6, 17, 7, 16, 10, 14, 12, 11, 13, 10, 13, 7, 12, 5, 10, 4, 7]], "7": [20, [17, 21, 7, 0, -1, -1, 3, 21, 17, 21]], "8": [20, [8, 21, 5, 20, 4, 18, 4, 16, 5, 14, 7, 13, 11, 12, 14, 11, 16, 9, 17, 7, 17, 4, 16, 2, 15, 1, 12, 0, 8, 0, 5, 1, 4, 2, 3, 4, 3, 7, 4, 9, 6, 11, 9, 12, 13, 13, 15, 14, 16, 16, 16, 18, 15, 20, 12, 21, 8, 21]], "9": [20, [16, 14, 15, 11, 13, 9, 10, 8, 9, 8, 6, 9, 4, 11, 3, 14, 3, 15, 4, 18, 6, 20, 9, 21, 10, 21, 13, 20, 15, 18, 16, 14, 16, 9, 15, 4, 13, 1, 10, 0, 8, 0, 5, 1, 4, 3]], ":": [10, [5, 14, 4, 13, 5, 12, 6, 13, 5, 14, -1, -1, 5, 2, 4, 1, 5, 0, 6, 1, 5, 2]], ";": [10, [5, 14, 4, 13, 5, 12, 6, 13, 5, 14, -1, -1, 6, 1, 5, 0, 4, 1, 5, 2, 6, 1, 6, -1, 5, -3, 4, -4]], "<": [24, [20, 18, 4, 9, 20, 0]], "=": [26, [4, 12, 22, 12, -1, -1, 4, 6, 22, 6]], ">": [24, [4, 18, 20, 9, 4, 0]], "?": [18, [3, 16, 3, 17, 4, 19, 5, 20, 7, 21, 11, 21, 13, 20, 14, 19, 15, 17, 15, 15, 14, 13, 13, 12, 9, 10, 9, 7, -1, -1, 9, 2, 8, 1, 9, 0, 10, 1, 9, 2]], "@": [27, [18, 13, 17, 15, 15, 16, 12, 16, 10, 15, 9, 14, 8, 11, 8, 8, 9, 6, 11, 5, 14, 5, 16, 6, 17, 8, -1, -1, 12, 16, 10, 14, 9, 11, 9, 8, 10, 6, 11, 5, -1, -1, 18, 16, 17, 8, 17, 6, 19, 5, 21, 5, 23, 7, 24, 10, 24, 12, 23, 15, 22, 17, 20, 19, 18, 20, 15, 21, 12, 21, 9, 20, 7, 19, 5, 17, 4, 15, 3, 12, 3, 9, 4, 6, 5, 4, 7, 2, 9, 1, 12, 0, 15, 0, 18, 1, 20, 2, 21, 3, -1, -1, 19, 16, 18, 8, 18, 6, 19, 5]], "A": [18, [9, 21, 1, 0, -1, -1, 9, 21, 17, 0, -1, -1, 4, 7, 14, 7]], "B": [21, [4, 21, 4, 0, -1, -1, 4, 21, 13, 21, 16, 20, 17, 19, 18, 17, 18, 15, 17, 13, 16, 12, 13, 11, -1, -1, 4, 11, 13, 11, 16, 10, 17, 9, 18, 7, 18, 4, 17, 2, 16, 1, 13, 0, 4, 0]], "C": [21, [18, 16, 17, 18, 15, 20, 13, 21, 9, 21, 7, 20, 5, 18, 4, 16, 3, 13, 3, 8, 4, 5, 5, 3, 7, 1, 9, 0, 13, 0, 15, 1, 17, 3, 18, 5]], "D": [21, [4, 21, 4, 0, -1, -1, 4, 21, 11, 21, 14, 20, 16, 18, 17, 16, 18, 13, 18, 8, 17, 5, 16, 3, 14, 1, 11, 0, 4, 0]], "E": [19, [4, 21, 4, 0, -1, -1, 4, 21, 17, 21, -1, -1, 4, 11, 12, 11, -1, -1, 4, 0, 17, 0]], "F": [18, [4, 21, 4, 0, -1, -1, 4, 21, 17, 21, -1, -1, 4, 11, 12, 11]], "G": [21, [18, 16, 17, 18, 15, 20, 13, 21, 9, 21, 7, 20, 5, 18, 4, 16, 3, 13, 3, 8, 4, 5, 5, 3, 7, 1, 9, 0, 13, 0, 15, 1, 17, 3, 18, 5, 18, 8, -1, -1, 13, 8, 18, 8]], "H": [22, [4, 21, 4, 0, -1, -1, 18, 21, 18, 0, -1, -1, 4, 11, 18, 11]], "I": [8, [4, 21, 4, 0]], "J": [16, [12, 21, 12, 5, 11, 2, 10, 1, 8, 0, 6, 0, 4, 1, 3, 2, 2, 5, 2, 7]], "K": [21, [4, 21, 4, 0, -1, -1, 18, 21, 4, 7, -1, -1, 9, 12, 18, 0]], "L": [17, [4, 21, 4, 0, -1, -1, 4, 0, 16, 0]], "M": [24, [4, 21, 4, 0, -1, -1, 4, 21, 12, 0, -1, -1, 20, 21, 12, 0, -1, -1, 20, 21, 20, 0]], "N": [22, [4, 21, 4, 0, -1, -1, 4, 21, 18, 0, -1, -1, 18, 21, 18, 0]], "O": [22, [9, 21, 7, 20, 5, 18, 4, 16, 3, 13, 3, 8, 4, 5, 5, 3, 7, 1, 9, 0, 13, 0, 15, 1, 17, 3, 18, 5, 19, 8, 19, 13, 18, 16, 17, 18, 15, 20, 13, 21, 9, 21]], "P": [21, [4, 21, 4, 0, -1, -1, 4, 21, 13, 21, 16, 20, 17, 19, 18, 17, 18, 14, 17, 12, 16, 11, 13, 10, 4, 10]], "Q": [22, [9, 21, 7, 20, 5, 18, 4, 16, 3, 13, 3, 8, 4, 5, 5, 3, 7, 1, 9, 0, 13, 0, 15, 1, 17, 3, 18, 5, 19, 8, 19, 13, 18, 16, 17, 18, 15, 20, 13, 21, 9, 21, -1, -1, 12, 4, 18, -2]], "R": [21, [4, 21, 4, 0, -1, -1, 4, 21, 13, 21, 16, 20, 17, 19, 18, 17, 18, 15, 17, 13, 16, 12, 13, 11, 4, 11, -1, -1, 11, 11, 18, 0]], "S": [20, [17, 18, 15, 20, 12, 21, 8, 21, 5, 20, 3, 18, 3, 16, 4, 14, 5, 13, 7, 12, 13, 10, 15, 9, 16, 8, 17, 6, 17, 3, 15, 1, 12, 0, 8, 0, 5, 1, 3, 3]], "T": [16, [8, 21, 8, 0, -1, -1, 1, 21, 15, 21]], "U": [22, [4, 21, 4, 6, 5, 3, 7, 1, 10, 0, 12, 0, 15, 1, 17, 3, 18, 6, 18, 21]], "V": [18, [1, 21, 9, 0, -1, -1, 17, 21, 9, 0]], "W": [24, [2, 21, 7, 0, -1, -1, 12, 21, 7, 0, -1, -1, 12, 21, 17, 0, -1, -1, 22, 21, 17, 0]], "X": [20, [3, 21, 17, 0, -1, -1, 17, 21, 3, 0]], "Y": [18, [1, 21, 9, 11, 9, 0, -1, -1, 17, 21, 9, 11]], "Z": [20, [17, 21, 3, 0, -1, -1, 3, 21, 17, 21, -1, -1, 3, 0, 17, 0]], "[": [14, [4, 25, 4, -7, -1, -1, 5, 25, 5, -7, -1, -1, 4, 25, 11, 25, -1, -1, 4, -7, 11, -7]], "\\": [14, [0, 21, 14, -3]], "]": [14, [9, 25, 9, -7, -1, -1, 10, 25, 10, -7, -1, -1, 3, 25, 10, 25, -1, -1, 3, -7, 10, -7]], "^": [16, [6, 15, 8, 18, 10, 15, -1, -1, 3, 12, 8, 17, 13, 12, -1, -1, 8, 17, 8, 0]], "_": [16, [0, -2, 16, -2]], "`": [10, [6, 21, 5, 20, 4, 18, 4, 16, 5, 15, 6, 16, 5, 17]], "a": [19, [15, 14, 15, 0, -1, -1, 15, 11, 13, 13, 11, 14, 8, 14, 6, 13, 4, 11, 3, 8, 3, 6, 4, 3, 6, 1, 8, 0, 11, 0, 13, 1, 15, 3]], "b": [19, [4, 21, 4, 0, -1, -1, 4, 11, 6, 13, 8, 14, 11, 14, 13, 13, 15, 11, 16, 8, 16, 6, 15, 3, 13, 1, 11, 0, 8, 0, 6, 1, 4, 3]], "c": [18, [15, 11, 13, 13, 11, 14, 8, 14, 6, 13, 4, 11, 3, 8, 3, 6, 4, 3, 6, 1, 8, 0, 11, 0, 13, 1, 15, 3]], "d": [19, [15, 21, 15, 0, -1, -1, 15, 11, 13, 13, 11, 14, 8, 14, 6, 13, 4, 11, 3, 8, 3, 6, 4, 3, 6, 1, 8, 0, 11, 0, 13, 1, 15, 3]], "e": [18, [3, 8, 15, 8, 15, 10, 14, 12, 13, 13, 11, 14, 8, 14, 6, 13, 4, 11, 3, 8, 3, 6, 4, 3, 6, 1, 8, 0, 11, 0, 13, 1, 15, 3]], "f": [12, [10, 21, 8, 21, 6, 20, 5, 17, 5, 0, -1, -1, 2, 14, 9, 14]], "g": [19, [15, 14, 15, -2, 14, -5, 13, -6, 11, -7, 8, -7, 6, -6, -1, -1, 15, 11, 13, 13, 11, 14, 8, 14, 6, 13, 4, 11, 3, 8, 3, 6, 4, 3, 6, 1, 8, 0, 11, 0, 13, 1, 15, 3]], "h": [19, [4, 21, 4, 0, -1, -1, 4, 10, 7, 13, 9, 14, 12, 14, 14, 13, 15, 10, 15, 0]], "i": [8, [3, 21, 4, 20, 5, 21, 4, 22, 3, 21, -1, -1, 4, 14, 4, 0]], "j": [10, [5, 21, 6, 20, 7, 21, 6, 22, 5, 21, -1, -1, 6, 14, 6, -3, 5, -6, 3, -7, 1, -7]], "k": [17, [4, 21, 4, 0, -1, -1, 14, 14, 4, 4, -1, -1, 8, 8, 15, 0]], "l": [8, [4, 21, 4, 0]], "m": [30, [4, 14, 4, 0, -1, -1, 4, 10, 7, 13, 9, 14, 12, 14, 14, 13, 15, 10, 15, 0, -1, -1, 15, 10, 18, 13, 20, 14, 23, 14, 25, 13, 26, 10, 26, 0]], "n": [19, [4, 14, 4, 0, -1, -1, 4, 10, 7, 13, 9, 14, 12, 14, 14, 13, 15, 10, 15, 0]], "o": [19, [8, 14, 6, 13, 4, 11, 3, 8, 3, 6, 4, 3, 6, 1, 8, 0, 11, 0, 13, 1, 15, 3, 16, 6, 16, 8, 15, 11, 13, 13, 11, 14, 8, 14]], "p": [19, [4, 14, 4, -7, -1, -1, 4, 11, 6, 13, 8, 14, 11, 14, 13, 13, 15, 11, 16, 8, 16, 6, 15, 3, 13, 1, 11, 0, 8, 0, 6, 1, 4, 3]], "q": [19, [15, 14, 15, -7, -1, -1, 15, 11, 13, 13, 11, 14, 8, 14, 6, 13, 4, 11, 3, 8, 3, 6, 4, 3, 6, 1, 8, 0, 11, 0, 13, 1, 15, 3]], "r": [13, [4, 14, 4, 0, -1, -1, 4, 8, 5, 11, 7, 13, 9, 14, 12, 14]], "s": [17, [14, 11, 13, 13, 10, 14, 7, 14, 4, 13, 3, 11, 4, 9, 6, 8, 11, 7, 13, 6, 14, 4, 14, 3, 13, 1, 10, 0, 7, 0, 4, 1, 3, 3]], "t": [12, [5, 21, 5, 4, 6, 1, 8, 0, 10, 0, -1, -1, 2, 14, 9, 14]], "u": [19, [4, 14, 4, 4, 5, 1, 7, 0, 10, 0, 12, 1, 15, 4, -1, -1, 15, 14, 15, 0]], "v": [16, [2, 14, 8, 0, -1, -1, 14, 14, 8, 0]], "w": [22, [3, 14, 7, 0, -1, -1, 11, 14, 7, 0, -1, -1, 11, 14, 15, 0, -1, -1, 19, 14, 15, 0]], "x": [17, [3, 14, 14, 0, -1, -1, 14, 14, 3, 0]], "y": [16, [2, 14, 8, 0, -1, -1, 14, 14, 8, 0, 6, -4, 4, -6, 2, -7, 1, -7]], "z": [17, [14, 14, 3, 0, -1, -1, 3, 14, 14, 14, -1, -1, 3, 0, 14, 0]], "{": [14, [9, 25, 7, 24, 6, 23, 5, 21, 5, 19, 6, 17, 7, 16, 8, 14, 8, 12, 6, 10, -1, -1, 7, 24, 6, 22, 6, 20, 7, 18, 8, 17, 9, 15, 9, 13, 8, 11, 4, 9, 8, 7, 9, 5, 9, 3, 8, 1, 7, 0, 6, -2, 6, -4, 7, -6, -1, -1, 6, 8, 8, 6, 8, 4, 7, 2, 6, 1, 5, -1, 5, -3, 6, -5, 7, -6, 9, -7]], "|": [8, [4, 25, 4, -7]], "}": [14, [5, 25, 7, 24, 8, 23, 9, 21, 9, 19, 8, 17, 7, 16, 6, 14, 6, 12, 8, 10, -1, -1, 7, 24, 8, 22, 8, 20, 7, 18, 6, 17, 5, 15, 5, 13, 6, 11, 10, 9, 6, 7, 5, 5, 5, 3, 6, 1, 7, 0, 8, -2, 8, -4, 7, -6, -1, -1, 8, 8, 6, 6, 6, 4, 7, 2, 8, 1, 9, -1, 9, -3, 8, -5, 7, -6, 5, -7]], "~": [24, [3, 6, 3, 8, 4, 11, 6, 12, 8, 12, 10, 11, 14, 8, 16, 7, 18, 7, 20, 8, 21, 10, -1, -1, 3, 8, 4, 10, 6, 11, 8, 11, 10, 10, 14, 7, 16, 6, 18, 6, 20, 7, 21, 10, 21, 12]], }; module.exports = function textVertices(text, left, baseline, scale) { scale = scale || 1; var strokes = [], i, len, j, len2, glyph, data, x, y, prev; for (i = 0, len = text.length; i < len; i++) { glyph = simplex_font[text[i]]; if (!glyph) continue; prev = null; for (j = 0, len2 = glyph[1].length; j < len2; j += 2) { if (glyph[1][j] === -1 && glyph[1][j + 1] === -1) { prev = null; } else { x = left + glyph[1][j] * scale; y = baseline - glyph[1][j + 1] * scale; if (prev) { strokes.push(prev.x, prev.y, x, y); } prev = {x: x, y: y}; } } left += glyph[0] * scale; } return strokes; }; },{}],21:[function(require,module,exports){ /** * @fileoverview gl-matrix - High performance matrix and vector operations * @author Brandon Jones * @author Colin MacKenzie IV * @version 2.2.0 */ /* Copyright (c) 2013, Brandon Jones, Colin MacKenzie IV. All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ (function(e){"use strict";var t={};typeof exports=="undefined"?typeof define=="function"&&typeof define.amd=="object"&&define.amd?(t.exports={},define(function(){return t.exports})):t.exports=typeof window!="undefined"?window:e:t.exports=exports,function(e){if(!t)var t=1e-6;if(!n)var n=typeof Float32Array!="undefined"?Float32Array:Array;if(!r)var r=Math.random;var i={};i.setMatrixArrayType=function(e){n=e},typeof e!="undefined"&&(e.glMatrix=i);var s={};s.create=function(){var e=new n(2);return e[0]=0,e[1]=0,e},s.clone=function(e){var t=new n(2);return t[0]=e[0],t[1]=e[1],t},s.fromValues=function(e,t){var r=new n(2);return r[0]=e,r[1]=t,r},s.copy=function(e,t){return e[0]=t[0],e[1]=t[1],e},s.set=function(e,t,n){return e[0]=t,e[1]=n,e},s.add=function(e,t,n){return e[0]=t[0]+n[0],e[1]=t[1]+n[1],e},s.subtract=function(e,t,n){return e[0]=t[0]-n[0],e[1]=t[1]-n[1],e},s.sub=s.subtract,s.multiply=function(e,t,n){return e[0]=t[0]*n[0],e[1]=t[1]*n[1],e},s.mul=s.multiply,s.divide=function(e,t,n){return e[0]=t[0]/n[0],e[1]=t[1]/n[1],e},s.div=s.divide,s.min=function(e,t,n){return e[0]=Math.min(t[0],n[0]),e[1]=Math.min(t[1],n[1]),e},s.max=function(e,t,n){return e[0]=Math.max(t[0],n[0]),e[1]=Math.max(t[1],n[1]),e},s.scale=function(e,t,n){return e[0]=t[0]*n,e[1]=t[1]*n,e},s.scaleAndAdd=function(e,t,n,r){return e[0]=t[0]+n[0]*r,e[1]=t[1]+n[1]*r,e},s.distance=function(e,t){var n=t[0]-e[0],r=t[1]-e[1];return Math.sqrt(n*n+r*r)},s.dist=s.distance,s.squaredDistance=function(e,t){var n=t[0]-e[0],r=t[1]-e[1];return n*n+r*r},s.sqrDist=s.squaredDistance,s.length=function(e){var t=e[0],n=e[1];return Math.sqrt(t*t+n*n)},s.len=s.length,s.squaredLength=function(e){var t=e[0],n=e[1];return t*t+n*n},s.sqrLen=s.squaredLength,s.negate=function(e,t){return e[0]=-t[0],e[1]=-t[1],e},s.normalize=function(e,t){var n=t[0],r=t[1],i=n*n+r*r;return i>0&&(i=1/Math.sqrt(i),e[0]=t[0]*i,e[1]=t[1]*i),e},s.dot=function(e,t){return e[0]*t[0]+e[1]*t[1]},s.cross=function(e,t,n){var r=t[0]*n[1]-t[1]*n[0];return e[0]=e[1]=0,e[2]=r,e},s.lerp=function(e,t,n,r){var i=t[0],s=t[1];return e[0]=i+r*(n[0]-i),e[1]=s+r*(n[1]-s),e},s.random=function(e,t){t=t||1;var n=r()*2*Math.PI;return e[0]=Math.cos(n)*t,e[1]=Math.sin(n)*t,e},s.transformMat2=function(e,t,n){var r=t[0],i=t[1];return e[0]=n[0]*r+n[2]*i,e[1]=n[1]*r+n[3]*i,e},s.transformMat2d=function(e,t,n){var r=t[0],i=t[1];return e[0]=n[0]*r+n[2]*i+n[4],e[1]=n[1]*r+n[3]*i+n[5],e},s.transformMat3=function(e,t,n){var r=t[0],i=t[1];return e[0]=n[0]*r+n[3]*i+n[6],e[1]=n[1]*r+n[4]*i+n[7],e},s.transformMat4=function(e,t,n){var r=t[0],i=t[1];return e[0]=n[0]*r+n[4]*i+n[12],e[1]=n[1]*r+n[5]*i+n[13],e},s.forEach=function(){var e=s.create();return function(t,n,r,i,s,o){var u,a;n||(n=2),r||(r=0),i?a=Math.min(i*n+r,t.length):a=t.length;for(u=r;u0&&(s=1/Math.sqrt(s),e[0]=t[0]*s,e[1]=t[1]*s,e[2]=t[2]*s),e},o.dot=function(e,t){return e[0]*t[0]+e[1]*t[1]+e[2]*t[2]},o.cross=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=n[0],u=n[1],a=n[2];return e[0]=i*a-s*u,e[1]=s*o-r*a,e[2]=r*u-i*o,e},o.lerp=function(e,t,n,r){var i=t[0],s=t[1],o=t[2];return e[0]=i+r*(n[0]-i),e[1]=s+r*(n[1]-s),e[2]=o+r*(n[2]-o),e},o.random=function(e,t){t=t||1;var n=r()*2*Math.PI,i=r()*2-1,s=Math.sqrt(1-i*i)*t;return e[0]=Math.cos(n)*s,e[1]=Math.sin(n)*s,e[2]=i*t,e},o.transformMat4=function(e,t,n){var r=t[0],i=t[1],s=t[2];return e[0]=n[0]*r+n[4]*i+n[8]*s+n[12],e[1]=n[1]*r+n[5]*i+n[9]*s+n[13],e[2]=n[2]*r+n[6]*i+n[10]*s+n[14],e},o.transformMat3=function(e,t,n){var r=t[0],i=t[1],s=t[2];return e[0]=r*n[0]+i*n[3]+s*n[6],e[1]=r*n[1]+i*n[4]+s*n[7],e[2]=r*n[2]+i*n[5]+s*n[8],e},o.transformQuat=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=n[0],u=n[1],a=n[2],f=n[3],l=f*r+u*s-a*i,c=f*i+a*r-o*s,h=f*s+o*i-u*r,p=-o*r-u*i-a*s;return e[0]=l*f+p*-o+c*-a-h*-u,e[1]=c*f+p*-u+h*-o-l*-a,e[2]=h*f+p*-a+l*-u-c*-o,e},o.forEach=function(){var e=o.create();return function(t,n,r,i,s,o){var u,a;n||(n=3),r||(r=0),i?a=Math.min(i*n+r,t.length):a=t.length;for(u=r;u0&&(o=1/Math.sqrt(o),e[0]=t[0]*o,e[1]=t[1]*o,e[2]=t[2]*o,e[3]=t[3]*o),e},u.dot=function(e,t){return e[0]*t[0]+e[1]*t[1]+e[2]*t[2]+e[3]*t[3]},u.lerp=function(e,t,n,r){var i=t[0],s=t[1],o=t[2],u=t[3];return e[0]=i+r*(n[0]-i),e[1]=s+r*(n[1]-s),e[2]=o+r*(n[2]-o),e[3]=u+r*(n[3]-u),e},u.random=function(e,t){return t=t||1,e[0]=r(),e[1]=r(),e[2]=r(),e[3]=r(),u.normalize(e,e),u.scale(e,e,t),e},u.transformMat4=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=t[3];return e[0]=n[0]*r+n[4]*i+n[8]*s+n[12]*o,e[1]=n[1]*r+n[5]*i+n[9]*s+n[13]*o,e[2]=n[2]*r+n[6]*i+n[10]*s+n[14]*o,e[3]=n[3]*r+n[7]*i+n[11]*s+n[15]*o,e},u.transformQuat=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=n[0],u=n[1],a=n[2],f=n[3],l=f*r+u*s-a*i,c=f*i+a*r-o*s,h=f*s+o*i-u*r,p=-o*r-u*i-a*s;return e[0]=l*f+p*-o+c*-a-h*-u,e[1]=c*f+p*-u+h*-o-l*-a,e[2]=h*f+p*-a+l*-u-c*-o,e},u.forEach=function(){var e=u.create();return function(t,n,r,i,s,o){var u,a;n||(n=4),r||(r=0),i?a=Math.min(i*n+r,t.length):a=t.length;for(u=r;u.999999?(r[0]=0,r[1]=0,r[2]=0,r[3]=1,r):(o.cross(e,i,s),r[0]=e[0],r[1]=e[1],r[2]=e[2],r[3]=1+u,h.normalize(r,r))}}(),h.setAxes=function(){var e=l.create();return function(t,n,r,i){return e[0]=r[0],e[3]=r[1],e[6]=r[2],e[1]=i[0],e[4]=i[1],e[7]=i[2],e[2]=n[0],e[5]=n[1],e[8]=n[2],h.normalize(t,h.fromMat3(t,e))}}(),h.clone=u.clone,h.fromValues=u.fromValues,h.copy=u.copy,h.set=u.set,h.identity=function(e){return e[0]=0,e[1]=0,e[2]=0,e[3]=1,e},h.setAxisAngle=function(e,t,n){n*=.5;var r=Math.sin(n);return e[0]=r*t[0],e[1]=r*t[1],e[2]=r*t[2],e[3]=Math.cos(n),e},h.add=u.add,h.multiply=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=t[3],u=n[0],a=n[1],f=n[2],l=n[3];return e[0]=r*l+o*u+i*f-s*a,e[1]=i*l+o*a+s*u-r*f,e[2]=s*l+o*f+r*a-i*u,e[3]=o*l-r*u-i*a-s*f,e},h.mul=h.multiply,h.scale=u.scale,h.rotateX=function(e,t,n){n*=.5;var r=t[0],i=t[1],s=t[2],o=t[3],u=Math.sin(n),a=Math.cos(n);return e[0]=r*a+o*u,e[1]=i*a+s*u,e[2]=s*a-i*u,e[3]=o*a-r*u,e},h.rotateY=function(e,t,n){n*=.5;var r=t[0],i=t[1],s=t[2],o=t[3],u=Math.sin(n),a=Math.cos(n);return e[0]=r*a-s*u,e[1]=i*a+o*u,e[2]=s*a+r*u,e[3]=o*a-i*u,e},h.rotateZ=function(e,t,n){n*=.5;var r=t[0],i=t[1],s=t[2],o=t[3],u=Math.sin(n),a=Math.cos(n);return e[0]=r*a+i*u,e[1]=i*a-r*u,e[2]=s*a+o*u,e[3]=o*a-s*u,e},h.calculateW=function(e,t){var n=t[0],r=t[1],i=t[2];return e[0]=n,e[1]=r,e[2]=i,e[3]=-Math.sqrt(Math.abs(1-n*n-r*r-i*i)),e},h.dot=u.dot,h.lerp=u.lerp,h.slerp=function(e,t,n,r){var i=t[0],s=t[1],o=t[2],u=t[3],a=n[0],f=n[1],l=n[2],c=n[3],h,p,d,v,m;return p=i*a+s*f+o*l+u*c,p<0&&(p=-p,a=-a,f=-f,l=-l,c=-c),1-p>1e-6?(h=Math.acos(p),d=Math.sin(h),v=Math.sin((1-r)*h)/d,m=Math.sin(r*h)/d):(v=1-r,m=r),e[0]=v*i+m*a,e[1]=v*s+m*f,e[2]=v*o+m*l,e[3]=v*u+m*c,e},h.invert=function(e,t){var n=t[0],r=t[1],i=t[2],s=t[3],o=n*n+r*r+i*i+s*s,u=o?1/o:0;return e[0]=-n*u,e[1]=-r*u,e[2]=-i*u,e[3]=s*u,e},h.conjugate=function(e,t){return e[0]=-t[0],e[1]=-t[1],e[2]=-t[2],e[3]=t[3],e},h.length=u.length,h.len=h.length,h.squaredLength=u.squaredLength,h.sqrLen=h.squaredLength,h.normalize=u.normalize,h.fromMat3=function(){var e=typeof Int8Array!="undefined"?new Int8Array([1,2,0]):[1,2,0];return function(t,n){var r=n[0]+n[4]+n[8],i;if(r>0)i=Math.sqrt(r+1),t[3]=.5*i,i=.5/i,t[0]=(n[7]-n[5])*i,t[1]=(n[2]-n[6])*i,t[2]=(n[3]-n[1])*i;else{var s=0;n[4]>n[0]&&(s=1),n[8]>n[s*3+s]&&(s=2);var o=e[s],u=e[o];i=Math.sqrt(n[s*3+s]-n[o*3+o]-n[u*3+u]+1),t[s]=.5*i,i=.5/i,t[3]=(n[u*3+o]-n[o*3+u])*i,t[o]=(n[o*3+s]+n[s*3+o])*i,t[u]=(n[u*3+s]+n[s*3+u])*i}return t}}(),h.str=function(e){return"quat("+e[0]+", "+e[1]+", "+e[2]+", "+e[3]+")"},typeof e!="undefined"&&(e.quat=h)}(t.exports)})(this); },{}],22:[function(require,module,exports){ 'use strict'; if (typeof window === 'undefined') { new (require('./source/worker.js'))(self); } else { // jshint -W079 var mapboxgl = module.exports = window.mapboxgl = {}; mapboxgl.Map = require('./ui/map.js'); mapboxgl.Navigation = require('./ui/control/navigation.js'); mapboxgl.Attribution = require('./ui/control/attribution.js'); mapboxgl.Source = require('./source/source'); mapboxgl.GeoJSONSource = require('./source/geojsonsource'); mapboxgl.VideoSource = require('./source/videosource'); mapboxgl.Style = require('./style/style.js'); mapboxgl.LatLng = require('./geo/latlng.js'); mapboxgl.LatLngBounds = require('./geo/latlngbounds.js'); mapboxgl.Point = require('point-geometry'); mapboxgl.Evented = require('./util/evented.js'); mapboxgl.util = require('./util/util.js'); var browser = require('./util/browser.js'); mapboxgl.util.supported = browser.supported; var ajax = require('./util/ajax.js'); mapboxgl.util.getJSON = ajax.getJSON; mapboxgl.util.getArrayBuffer = ajax.getArrayBuffer; var config = require('./util/config.js'); mapboxgl.config = config; Object.defineProperty(mapboxgl, 'accessToken', { get: function() { return config.ACCESS_TOKEN; }, set: function(token) { config.ACCESS_TOKEN = token; } }); } },{"./geo/latlng.js":17,"./geo/latlngbounds.js":18,"./source/geojsonsource":36,"./source/source":40,"./source/videosource":45,"./source/worker.js":46,"./style/style.js":53,"./ui/control/attribution.js":67,"./ui/control/navigation.js":69,"./ui/map.js":74,"./util/ajax.js":76,"./util/browser.js":77,"./util/config.js":80,"./util/evented.js":82,"./util/util.js":87,"point-geometry":98}],23:[function(require,module,exports){ 'use strict'; var mat3 = require('../lib/glmatrix.js').mat3; module.exports = drawBackground; function drawBackground(gl, painter, bucket, layerStyle, posMatrix, params, imageSprite) { var color = layerStyle['background-color']; var image = layerStyle['background-image']; var opacity = layerStyle['background-opacity'] || 1; var shader; if (image) { // Draw texture fill var imagePos = imageSprite.getPosition(image, true); if (!imagePos) return; shader = painter.patternShader; gl.switchShader(shader, posMatrix); gl.uniform1i(shader.u_image, 0); gl.uniform2fv(shader.u_pattern_tl, imagePos.tl); gl.uniform2fv(shader.u_pattern_br, imagePos.br); gl.uniform1f(shader.u_mix, painter.transform.zoomFraction); gl.uniform1f(shader.u_opacity, opacity); var transform = painter.transform; var size = imagePos.size; var center = transform.locationCoordinate(transform.center); var scale = 1 / Math.pow(2, transform.zoomFraction); var matrix = mat3.create(); mat3.scale(matrix, matrix, [1 / size[0], 1 / size[1], 1]); mat3.translate(matrix, matrix, [ (center.column * transform.tileSize) % size[0], (center.row * transform.tileSize) % size[1], 0 ]); mat3.rotate(matrix, matrix, -transform.angle); mat3.scale(matrix, matrix, [ scale * transform.width / 2, -scale * transform.height / 2, 1 ]); gl.uniformMatrix3fv(shader.u_patternmatrix, false, matrix); imageSprite.bind(gl, true); } else { // Draw filling rectangle. shader = painter.fillShader; gl.switchShader(shader, params.padded || posMatrix); gl.uniform4fv(shader.u_color, color); } gl.disable(gl.STENCIL_TEST); gl.bindBuffer(gl.ARRAY_BUFFER, painter.backgroundBuffer); gl.vertexAttribPointer(shader.a_pos, painter.backgroundBuffer.itemSize, gl.SHORT, false, 0, 0); gl.drawArrays(gl.TRIANGLE_STRIP, 0, painter.backgroundBuffer.itemCount); gl.enable(gl.STENCIL_TEST); gl.stencilMask(0x00); gl.stencilFunc(gl.EQUAL, 0x80, 0x80); } },{"../lib/glmatrix.js":21}],24:[function(require,module,exports){ 'use strict'; module.exports = drawComposited; function drawComposited (gl, painter, buckets, layerStyle, params, style, layer) { var texture = painter.namedRenderTextures[layer.id]; if (!texture) return console.warn('missing render texture ' + layer.id); gl.disable(gl.STENCIL_TEST); gl.stencilMask(0x00); gl.switchShader(painter.compositeShader, painter.identityMatrix); gl.activeTexture(gl.TEXTURE0); gl.bindTexture(gl.TEXTURE_2D, texture); gl.uniform1i(painter.compositeShader.u_image, 0); gl.uniform1f(painter.compositeShader.u_opacity, layerStyle['composite-opacity']); gl.bindBuffer(gl.ARRAY_BUFFER, painter.backgroundBuffer); gl.vertexAttribPointer(painter.compositeShader.a_pos, 2, gl.SHORT, false, 0, 0); gl.drawArrays(gl.TRIANGLE_STRIP, 0, 4); gl.enable(gl.STENCIL_TEST); painter.freeRenderTexture(layer.id); } },{}],25:[function(require,module,exports){ 'use strict'; var textVertices = require('../lib/debugtext.js'); var browser = require('../util/browser.js'); module.exports = drawDebug; function drawDebug(gl, painter, tile, params) { // Blend to the front, not the back. gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA); gl.switchShader(painter.debugShader, painter.tile.posMatrix, painter.tile.exMatrix); // draw bounding rectangle gl.bindBuffer(gl.ARRAY_BUFFER, painter.debugBuffer); gl.vertexAttribPointer(painter.debugShader.a_pos, painter.debugBuffer.itemSize, gl.SHORT, false, 0, 0); gl.uniform4f(painter.debugShader.u_color, 1, 0, 0, 1); gl.lineWidth(4); gl.drawArrays(gl.LINE_STRIP, 0, painter.debugBuffer.itemCount); // draw tile coordinate var coord = params.z + '/' + params.x + '/' + params.y; var vertices = textVertices(coord, 50, 200, 5); gl.bindBuffer(gl.ARRAY_BUFFER, painter.debugTextBuffer); gl.bufferData(gl.ARRAY_BUFFER, new Int16Array(vertices), gl.STREAM_DRAW); gl.vertexAttribPointer(painter.debugShader.a_pos, painter.debugTextBuffer.itemSize, gl.SHORT, false, 0, 0); gl.lineWidth(8 * browser.devicePixelRatio); gl.uniform4f(painter.debugShader.u_color, 1, 1, 1, 1); gl.drawArrays(gl.LINES, 0, vertices.length / painter.debugTextBuffer.itemSize); gl.lineWidth(2 * browser.devicePixelRatio); gl.uniform4f(painter.debugShader.u_color, 0, 0, 0, 1); gl.drawArrays(gl.LINES, 0, vertices.length / painter.debugTextBuffer.itemSize); // Revert blending mode to blend to the back. gl.blendFunc(gl.ONE_MINUS_DST_ALPHA, gl.ONE); } },{"../lib/debugtext.js":20,"../util/browser.js":77}],26:[function(require,module,exports){ 'use strict'; var browser = require('../util/browser.js'); var mat3 = require('../lib/glmatrix.js').mat3; module.exports = drawFill; function drawFill(gl, painter, bucket, layerStyle, posMatrix, params, imageSprite) { var translatedPosMatrix = painter.translateMatrix(posMatrix, params.z, layerStyle['fill-translate'], layerStyle['fill-translate-anchor']); var color = layerStyle['fill-color']; var vertex, elements, group, count; // Draw the stencil mask. // We're only drawing to the first seven bits (== support a maximum of // 127 overlapping polygons in one place before we get rendering errors). gl.stencilMask(0x3F); gl.clear(gl.STENCIL_BUFFER_BIT); // Draw front facing triangles. Wherever the 0x80 bit is 1, we are // increasing the lower 7 bits by one if the triangle is a front-facing // triangle. This means that all visible polygons should be in CCW // orientation, while all holes (see below) are in CW orientation. gl.stencilFunc(gl.NOTEQUAL, 0x80, 0x80); // When we do a nonzero fill, we count the number of times a pixel is // covered by a counterclockwise polygon, and subtract the number of // times it is "uncovered" by a clockwise polygon. gl.stencilOpSeparate(gl.FRONT, gl.INCR_WRAP, gl.KEEP, gl.KEEP); gl.stencilOpSeparate(gl.BACK, gl.DECR_WRAP, gl.KEEP, gl.KEEP); // When drawing a shape, we first draw all shapes to the stencil buffer // and incrementing all areas where polygons are gl.colorMask(false, false, false, false); // Draw the actual triangle fan into the stencil buffer. gl.switchShader(painter.fillShader, translatedPosMatrix, painter.tile.exMatrix); // Draw all buffers vertex = bucket.buffers.fillVertex; vertex.bind(gl); elements = bucket.buffers.fillElement; elements.bind(gl); var offset, elementOffset; for (var i = 0; i < bucket.elementGroups.groups.length; i++) { group = bucket.elementGroups.groups[i]; offset = group.vertexStartIndex * vertex.itemSize; gl.vertexAttribPointer(painter.fillShader.a_pos, 2, gl.SHORT, false, 4, offset + 0); count = group.elementLength * 3; elementOffset = group.elementStartIndex * elements.itemSize; gl.drawElements(gl.TRIANGLES, count, gl.UNSIGNED_SHORT, elementOffset); } // Now that we have the stencil mask in the stencil buffer, we can start // writing to the color buffer. gl.colorMask(true, true, true, true); // From now on, we don't want to update the stencil buffer anymore. gl.stencilOp(gl.KEEP, gl.KEEP, gl.KEEP); gl.stencilMask(0x0); var strokeColor = layerStyle['fill-outline-color']; // Because we're drawing top-to-bottom, and we update the stencil mask // below, we have to draw the outline first (!) if (layerStyle['fill-antialias'] === true && params.antialiasing && !(layerStyle['fill-image'] && !strokeColor)) { gl.switchShader(painter.outlineShader, translatedPosMatrix, painter.tile.exMatrix); gl.lineWidth(2 * browser.devicePixelRatio); if (strokeColor) { // If we defined a different color for the fill outline, we are // going to ignore the bits in 0x3F and just care about the global // clipping mask. gl.stencilFunc(gl.EQUAL, 0x80, 0x80); } else { // Otherwise, we only want to draw the antialiased parts that are // *outside* the current shape. This is important in case the fill // or stroke color is translucent. If we wouldn't clip to outside // the current shape, some pixels from the outline stroke overlapped // the (non-antialiased) fill. gl.stencilFunc(gl.EQUAL, 0x80, 0xBF); } gl.uniform2f(painter.outlineShader.u_world, gl.drawingBufferWidth, gl.drawingBufferHeight); gl.uniform4fv(painter.outlineShader.u_color, strokeColor ? strokeColor : color); // Draw all buffers vertex = bucket.buffers.fillVertex; elements = bucket.buffers.outlineElement; elements.bind(gl); for (var k = 0; k < bucket.elementGroups.groups.length; k++) { group = bucket.elementGroups.groups[k]; offset = group.vertexStartIndex * vertex.itemSize; gl.vertexAttribPointer(painter.outlineShader.a_pos, 2, gl.SHORT, false, 4, offset + 0); count = group.secondElementLength * 2; elementOffset = group.secondElementStartIndex * elements.itemSize; gl.drawElements(gl.LINES, count, gl.UNSIGNED_SHORT, elementOffset); } } var image = layerStyle['fill-image']; var opacity = layerStyle['fill-opacity'] || 1; var shader; if (image) { // Draw texture fill var imagePos = imageSprite.getPosition(image, true); if (!imagePos) return; shader = painter.patternShader; gl.switchShader(shader, posMatrix); gl.uniform1i(shader.u_image, 0); gl.uniform2fv(shader.u_pattern_tl, imagePos.tl); gl.uniform2fv(shader.u_pattern_br, imagePos.br); gl.uniform1f(shader.u_mix, painter.transform.zoomFraction); gl.uniform1f(shader.u_opacity, opacity); var factor = 8 / Math.pow(2, painter.transform.tileZoom - params.z); var matrix = mat3.create(); mat3.scale(matrix, matrix, [ 1 / (imagePos.size[0] * factor), 1 / (imagePos.size[1] * factor), 1, 1 ]); gl.uniformMatrix3fv(shader.u_patternmatrix, false, matrix); imageSprite.bind(gl, true); } else { // Draw filling rectangle. shader = painter.fillShader; gl.switchShader(shader, params.padded || posMatrix); gl.uniform4fv(shader.u_color, color); } // Only draw regions that we marked gl.stencilFunc(gl.NOTEQUAL, 0x0, 0x3F); gl.bindBuffer(gl.ARRAY_BUFFER, painter.tileExtentBuffer); gl.vertexAttribPointer(shader.a_pos, painter.tileExtentBuffer.itemSize, gl.SHORT, false, 0, 0); gl.drawArrays(gl.TRIANGLE_STRIP, 0, painter.tileExtentBuffer.itemCount); gl.stencilMask(0x00); gl.stencilFunc(gl.EQUAL, 0x80, 0x80); } },{"../lib/glmatrix.js":21,"../util/browser.js":77}],27:[function(require,module,exports){ 'use strict'; var browser = require('../util/browser.js'); module.exports = function drawLine(gl, painter, bucket, layerStyle, posMatrix, params, imageSprite) { posMatrix = painter.translateMatrix(posMatrix, params.z, layerStyle['line-translate'], layerStyle['line-translate-anchor']); // don't draw zero-width lines if (layerStyle['line-width'] <= 0) return; var gamma = 1; var antialiasing = gamma / browser.devicePixelRatio; var lineOffset = layerStyle['line-offset'] / 2; var inset = Math.max(-1, lineOffset - layerStyle['line-width'] / 2 - antialiasing / 2) + 1; var outset = lineOffset + layerStyle['line-width'] / 2 + antialiasing / 2; var imagePos = layerStyle['line-image'] && imageSprite.getPosition(layerStyle['line-image']); var shader; if (imagePos) { var factor = 8 / Math.pow(2, painter.transform.tileZoom - params.z); imageSprite.bind(gl, true); //factor = Math.pow(2, 4 - painter.transform.tileZoom + params.z); shader = painter.linepatternShader; gl.switchShader(shader, posMatrix, painter.tile.exMatrix); gl.uniform2fv(shader.u_pattern_size, [imagePos.size[0] * factor, imagePos.size[1] ]); gl.uniform2fv(shader.u_pattern_tl, imagePos.tl); gl.uniform2fv(shader.u_pattern_br, imagePos.br); gl.uniform1f(shader.u_fade, painter.transform.zoomFraction); } else { shader = painter.lineShader; gl.switchShader(shader, posMatrix, painter.tile.exMatrix); gl.uniform2fv(shader.u_dasharray, layerStyle['line-dasharray']); gl.uniform4fv(shader.u_color, layerStyle['line-color']); } var tilePixelRatio = painter.transform.scale / (1 << params.z) / 8; gl.uniform2fv(shader.u_linewidth, [ outset, inset ]); gl.uniform1f(shader.u_ratio, tilePixelRatio); gl.uniform1f(shader.u_blur, layerStyle['line-blur'] + antialiasing); var vertex = bucket.buffers.lineVertex; vertex.bind(gl); var element = bucket.buffers.lineElement; element.bind(gl); var groups = bucket.elementGroups.groups; for (var i = 0; i < groups.length; i++) { var group = groups[i]; var offset = group.vertexStartIndex * vertex.itemSize; gl.vertexAttribPointer(shader.a_pos, 4, gl.SHORT, false, 8, offset + 0); gl.vertexAttribPointer(shader.a_extrude, 2, gl.BYTE, false, 8, offset + 6); gl.vertexAttribPointer(shader.a_linesofar, 2, gl.SHORT, false, 8, offset + 4); var count = group.elementLength * 3; var elementOffset = group.elementStartIndex * element.itemSize; gl.drawElements(gl.TRIANGLES, count, gl.UNSIGNED_SHORT, elementOffset); } }; },{"../util/browser.js":77}],28:[function(require,module,exports){ 'use strict'; var TileCoord = require('../source/tilecoord.js'); var PrerenderedTexture = require('./prerendered.js'); var mat4 = require('../lib/glmatrix.js').mat4; module.exports = drawRaster; function drawRaster(gl, painter, bucket, layerStyle, params, style, layer, tile) { if (layer && layer.layers) { if (!bucket.prerendered) { bucket.prerendered = new PrerenderedTexture(gl, bucket.info, painter); bucket.prerendered.bindFramebuffer(); gl.clearStencil(0x80); gl.stencilMask(0xFF); gl.clear(gl.STENCIL_BUFFER_BIT | gl.COLOR_BUFFER_BIT); gl.stencilMask(0x00); gl.viewport(0, 0, bucket.prerendered.size, bucket.prerendered.size); var buffer = bucket.prerendered.buffer * 4096; var matrix = mat4.create(); mat4.ortho(matrix, -buffer, 4096 + buffer, -4096 - buffer, buffer, 0, 1); mat4.translate(matrix, matrix, [0, -4096, 0]); params.padded = mat4.create(); mat4.ortho(params.padded, 0, 4096, -4096, 0, 0, 1); mat4.translate(params.padded, params.padded, [0, -4096, 0]); painter.draw(tile, style, layer.layers, params, matrix); delete params.padded; if (bucket.info['raster-blur'] > 0) { bucket.prerendered.blur(painter, bucket.info['raster-blur']); } bucket.prerendered.unbindFramebuffer(); gl.viewport(0, 0, painter.width, painter.height); } } var texture = bucket.tile ? bucket.tile : bucket.prerendered; gl.disable(gl.STENCIL_TEST); var shader = painter.rasterShader; gl.switchShader(shader, painter.tile.posMatrix, painter.tile.exMatrix); // color parameters gl.uniform1f(shader.u_brightness_low, layerStyle['raster-brightness'][0]); gl.uniform1f(shader.u_brightness_high, layerStyle['raster-brightness'][1]); gl.uniform1f(shader.u_saturation_factor, saturationFactor(layerStyle['raster-saturation'])); gl.uniform1f(shader.u_contrast_factor, contrastFactor(layerStyle['raster-contrast'])); gl.uniform3fv(shader.u_spin_weights, spinWeights(layerStyle['raster-hue-rotate'])); var parentTile, opacities; if (layer && layer.layers) { parentTile = null; opacities = [layerStyle['raster-opacity'], 0]; } else { parentTile = findParent(texture); opacities = getOpacities(texture, parentTile); } var parentScaleBy, parentTL; gl.activeTexture(gl.TEXTURE0); texture.bind(gl); if (parentTile) { gl.activeTexture(gl.TEXTURE1); parentTile.bind(gl); var tilePos = TileCoord.fromID(texture.id); var parentPos = parentTile && TileCoord.fromID(parentTile.id); parentScaleBy = Math.pow(2, parentPos.z - tilePos.z); parentTL = [tilePos.x * parentScaleBy % 1, tilePos.y * parentScaleBy % 1]; } else { opacities[1] = 0; } var bufferScale = bucket.prerendered ? (4096 * (1 + 2 * bucket.prerendered.buffer)) / 4096 : 1; // cross-fade parameters gl.uniform2fv(shader.u_tl_parent, parentTL || [0, 0]); gl.uniform1f(shader.u_scale_parent, parentScaleBy || 1); gl.uniform1f(shader.u_buffer_scale, bufferScale); gl.uniform1f(shader.u_opacity0, opacities[0]); gl.uniform1f(shader.u_opacity1, opacities[1]); gl.uniform1i(shader.u_image0, 0); gl.uniform1i(shader.u_image1, 1); gl.bindBuffer(gl.ARRAY_BUFFER, texture.boundsBuffer || painter.tileExtentBuffer); gl.vertexAttribPointer(shader.a_pos, 2, gl.SHORT, false, 8, 0); gl.vertexAttribPointer(shader.a_texture_pos, 2, gl.SHORT, false, 8, 4); gl.drawArrays(gl.TRIANGLE_STRIP, 0, 4); gl.enable(gl.STENCIL_TEST); } function findParent(tile) { var source = tile.source; if (!source) return; var parentTiles = {}; source._findLoadedParent(tile.id, source.options.minZoom, parentTiles); return source.tiles[Object.keys(parentTiles)[0]]; } function clamp(n, min, max) { return Math.max(min, Math.min(max, n)); } function spinWeights(angle) { angle *= Math.PI / 180; var s = Math.sin(angle); var c = Math.cos(angle); return [ (2 * c + 1) / 3, (-Math.sqrt(3) * s - c + 1) / 3, (Math.sqrt(3) * s - c + 1) / 3 ]; } function contrastFactor(contrast) { return contrast > 0 ? 1 / (1 - contrast) : 1 + contrast; } function saturationFactor(saturation) { return saturation > 0 ? 1 - 1 / (1.001 - saturation) : -saturation; } function getOpacities(tile, parentTile) { if (!tile.source) return [1, 0]; var now = new Date().getTime(); var fadeDuration = tile.source.map.style.rasterFadeDuration; var sinceTile = (now - tile.timeAdded) / fadeDuration; var sinceParent = parentTile ? (now - parentTile.timeAdded) / fadeDuration : -1; var tilePos = TileCoord.fromID(tile.id); var parentPos = parentTile && TileCoord.fromID(parentTile.id); var idealZ = tile.source._coveringZoomLevel(tile.source._getZoom()); var parentFurther = parentTile ? Math.abs(parentPos.z - idealZ) > Math.abs(tilePos.z - idealZ) : false; var opacity = []; if (!parentTile || parentFurther) { // if no parent or parent is older opacity[0] = clamp(sinceTile, 0, 1); opacity[1] = 1 - opacity[0]; } else { // parent is younger, zooming out opacity[0] = clamp(1 - sinceParent, 0, 1); opacity[1] = 1 - opacity[0]; } return opacity; } },{"../lib/glmatrix.js":21,"../source/tilecoord.js":42,"./prerendered.js":34}],29:[function(require,module,exports){ 'use strict'; var browser = require('../util/browser.js'); var mat4 = require('../lib/glmatrix.js').mat4; module.exports = drawSymbols; function drawSymbols(gl, painter, bucket, layerStyle, posMatrix, params, imageSprite) { gl.disable(gl.STENCIL_TEST); if (bucket.elementGroups.text.groups.length) { drawSymbol(gl, painter, bucket, layerStyle, posMatrix, params, imageSprite, 'text'); } if (bucket.elementGroups.icon.groups.length) { drawSymbol(gl, painter, bucket, layerStyle, posMatrix, params, imageSprite, 'icon'); } gl.enable(gl.STENCIL_TEST); } var defaultSizes = { icon: 1, text: 24 }; function drawSymbol(gl, painter, bucket, layerStyle, posMatrix, params, imageSprite, prefix) { posMatrix = painter.translateMatrix(posMatrix, params.z, layerStyle[prefix + '-translate'], layerStyle[prefix + '-translate-anchor']); var info = bucket.info; var exMatrix = mat4.clone(painter.projectionMatrix); var alignedWithMap = info[prefix + '-rotation-alignment'] === 'map'; var angleOffset = (alignedWithMap ? painter.transform.angle : 0); if (angleOffset) { mat4.rotateZ(exMatrix, exMatrix, angleOffset); } // If layerStyle.size > info[prefix + '-max-size'] then labels may collide var fontSize = layerStyle[prefix + '-size'] || info[prefix + '-max-size']; var fontScale = fontSize / defaultSizes[prefix]; mat4.scale(exMatrix, exMatrix, [ fontScale, fontScale, 1 ]); var text = prefix === 'text'; var sdf = text || bucket.elementGroups.sdfIcons; var shader, buffer, texsize; if (!text && !imageSprite.loaded()) return; gl.activeTexture(gl.TEXTURE0); if (sdf) { shader = painter.sdfShader; } else { shader = painter.iconShader; } if (text) { painter.glyphAtlas.updateTexture(gl); buffer = bucket.buffers.glyphVertex; texsize = [painter.glyphAtlas.width / 4, painter.glyphAtlas.height / 4]; } else { imageSprite.bind(gl, alignedWithMap || params.rotating || params.zooming || fontScale != 1 || sdf); buffer = bucket.buffers.iconVertex; texsize = [imageSprite.img.width, imageSprite.img.height]; } gl.switchShader(shader, posMatrix, exMatrix); gl.uniform1i(shader.u_texture, 0); gl.uniform2fv(shader.u_texsize, texsize); buffer.bind(gl); var ubyte = gl.UNSIGNED_BYTE; var stride = text ? 16 : 20; gl.vertexAttribPointer(shader.a_pos, 2, gl.SHORT, false, stride, 0); gl.vertexAttribPointer(shader.a_offset, 2, gl.SHORT, false, stride, 4); gl.vertexAttribPointer(shader.a_labelminzoom, 1, ubyte, false, stride, 8); gl.vertexAttribPointer(shader.a_minzoom, 1, ubyte, false, stride, 9); gl.vertexAttribPointer(shader.a_maxzoom, 1, ubyte, false, stride, 10); gl.vertexAttribPointer(shader.a_angle, 1, ubyte, false, stride, 11); gl.vertexAttribPointer(shader.a_rangeend, 1, ubyte, false, stride, 12); gl.vertexAttribPointer(shader.a_rangestart, 1, ubyte, false, stride, 13); if (text) { gl.vertexAttribPointer(shader.a_tex, 2, ubyte, false, stride, 14); } else { gl.vertexAttribPointer(shader.a_tex, 2, gl.SHORT, false, stride, 16); } // Convert the -pi..pi to an int8 range. var angle = Math.round((painter.transform.angle) / Math.PI * 128); // adjust min/max zooms for variable font sies var zoomAdjust = Math.log(fontSize / info[prefix + '-max-size']) / Math.LN2 || 0; var flip = alignedWithMap && info[prefix + '-keep-upright']; gl.uniform1f(shader.u_flip, flip ? 1 : 0); gl.uniform1f(shader.u_angle, (angle + 256) % 256); gl.uniform1f(shader.u_zoom, (painter.transform.zoom - zoomAdjust) * 10); // current zoom level var f = painter.frameHistory.getFadeProperties(300); gl.uniform1f(shader.u_fadedist, f.fadedist * 10); gl.uniform1f(shader.u_minfadezoom, Math.floor(f.minfadezoom * 10)); gl.uniform1f(shader.u_maxfadezoom, Math.floor(f.maxfadezoom * 10)); gl.uniform1f(shader.u_fadezoom, (painter.transform.zoom + f.bump) * 10); if (!sdf) gl.uniform1f(shader.u_opacity, layerStyle['icon-opacity']); var sdfFontSize = text ? 24 : 1; var sdfPx = 8; var blurOffset = 1.19; var haloOffset = 6; if (sdf) { gl.uniform1f(shader.u_gamma, 0.105 * sdfFontSize / fontSize / browser.devicePixelRatio); gl.uniform4fv(shader.u_color, layerStyle[prefix + '-color']); gl.uniform1f(shader.u_buffer, (256 - 64) / 256); } var begin = bucket.elementGroups[prefix].groups[0].vertexStartIndex, len = bucket.elementGroups[prefix].groups[0].vertexLength; gl.drawArrays(gl.TRIANGLES, begin, len); if (sdf && layerStyle[prefix + '-halo-color']) { // Draw halo underneath the text. gl.uniform1f(shader.u_gamma, (layerStyle[prefix + '-halo-blur'] * blurOffset / (fontSize / sdfFontSize) / sdfPx) + (0.105 * sdfFontSize / fontSize) / browser.devicePixelRatio); gl.uniform4fv(shader.u_color, layerStyle[prefix + '-halo-color']); gl.uniform1f(shader.u_buffer, (haloOffset - layerStyle[prefix + '-halo-width'] / (fontSize / sdfFontSize)) / sdfPx); gl.drawArrays(gl.TRIANGLES, begin, len); } } },{"../lib/glmatrix.js":21,"../util/browser.js":77}],30:[function(require,module,exports){ 'use strict'; var browser = require('../util/browser.js'); var mat4 = require('../lib/glmatrix.js').mat4; module.exports = drawVertices; function drawVertices(gl, painter, bucket) { // Blend to the front, not the back. gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA); gl.switchShader(painter.dotShader, painter.tile.posMatrix, painter.tile.exMatrix); // // Draw debug points. gl.uniform1f(painter.dotShader.u_size, 4 * browser.devicePixelRatio); gl.uniform1f(painter.dotShader.u_blur, 0.25); gl.uniform4fv(painter.dotShader.u_color, [0.25, 0, 0, 0.25]); // Draw the actual triangle fan into the stencil buffer. var vertex, groups, group, begin, count; // Draw all buffers if (bucket.info.fill) { vertex = bucket.buffers.fillVertex; vertex.bind(gl); groups = bucket.elementGroups.groups; for (var i = 0; i < groups.length; i++) { group = groups[i]; begin = group.vertexStartIndex; count = group.vertexLength; gl.vertexAttribPointer(painter.dotShader.a_pos, 2, gl.SHORT, false, 0, 0); gl.drawArrays(gl.POINTS, begin, count); } } var newPosMatrix = mat4.clone(painter.tile.posMatrix); mat4.scale(newPosMatrix, newPosMatrix, [0.5, 0.5, 1]); gl.switchShader(painter.dotShader, newPosMatrix, painter.tile.exMatrix); // Draw all line buffers if (bucket.info.line) { vertex = bucket.buffers.lineVertex; vertex.bind(gl); groups = bucket.elementGroups.groups; for (var k = 0; k < groups.length; k++) { group = groups[k]; begin = group.vertexStartIndex; count = group.vertexLength; gl.vertexAttribPointer(painter.dotShader.a_pos, 2, gl.SHORT, false, 0, 0); gl.drawArrays(gl.POINTS, begin, count); } } // Revert blending mode to blend to the back. gl.blendFunc(gl.ONE_MINUS_DST_ALPHA, gl.ONE); } },{"../lib/glmatrix.js":21,"../util/browser.js":77}],31:[function(require,module,exports){ 'use strict'; module.exports = FrameHistory; function FrameHistory() { } FrameHistory.prototype.getFadeProperties = function(duration) { if (duration === undefined) duration = 300; var currentTime = (new Date()).getTime(); // Remove frames until only one is outside the duration, or until there are only three while (frameHistory.length > 3 && frameHistory[1].time + duration < currentTime) { frameHistory.shift(); } if (frameHistory[1].time + duration < currentTime) { frameHistory[0].z = frameHistory[1].z; } var frameLen = frameHistory.length; if (frameLen < 3) console.warn('there should never be less than three frames in the history'); // Find the range of zoom levels we want to fade between var startingZ = frameHistory[0].z, lastFrame = frameHistory[frameLen - 1], endingZ = lastFrame.z, lowZ = Math.min(startingZ, endingZ), highZ = Math.max(startingZ, endingZ); // Calculate the speed of zooming, and how far it would zoom in terms of zoom levels in one duration var zoomDiff = lastFrame.z - frameHistory[1].z, timeDiff = lastFrame.time - frameHistory[1].time; var fadedist = zoomDiff / (timeDiff / duration); if (isNaN(fadedist)) console.warn('fadedist should never be NaN'); // At end of a zoom when the zoom stops changing continue pretending to zoom at that speed // bump is how much farther it would have been if it had continued zooming at the same rate var bump = (currentTime - lastFrame.time) / duration * fadedist; return { fadedist: fadedist, minfadezoom: lowZ, maxfadezoom: highZ, bump: bump }; }; // Store previous render times var frameHistory = []; // Record frame history that will be used to calculate fading params FrameHistory.prototype.record = function(zoom) { var currentTime = (new Date()).getTime(); // first frame ever if (!frameHistory.length) { frameHistory.push({time: 0, z: zoom }, {time: 0, z: zoom }); } if (frameHistory.length === 2 || frameHistory[frameHistory.length - 1].z !== zoom) { frameHistory.push({ time: currentTime, z: zoom }); } }; },{}],32:[function(require,module,exports){ 'use strict'; var shaders = require('./shaders.js'); exports.extend = function(context) { context.getShader = function(name, type) { var kind = type == this.FRAGMENT_SHADER ? 'fragment' : 'vertex'; if (!shaders[name] || !shaders[name][kind]) { throw new Error("Could not find shader " + name); } var shader = this.createShader(type); this.shaderSource(shader, shaders[name][kind]); this.compileShader(shader); if (!this.getShaderParameter(shader, this.COMPILE_STATUS)) { throw new Error(this.getShaderInfoLog(shader)); } return shader; }; context.initializeShader = function(name, attributes, uniforms) { var shader = { program: this.createProgram(), fragment: this.getShader(name, this.FRAGMENT_SHADER), vertex: this.getShader(name, this.VERTEX_SHADER), attributes: [] }; this.attachShader(shader.program, shader.vertex); this.attachShader(shader.program, shader.fragment); this.linkProgram(shader.program); if (!this.getProgramParameter(shader.program, this.LINK_STATUS)) { console.error(this.getProgramInfoLog(shader.program)); alert("Could not initialize shader " + name); } else { for (var i = 0; i < attributes.length; i++) { shader[attributes[i]] = this.getAttribLocation(shader.program, attributes[i]); shader.attributes.push(shader[attributes[i]]); } for (var k = 0; k < uniforms.length; k++) { shader[uniforms[k]] = this.getUniformLocation(shader.program, uniforms[k]); } } return shader; }; // Switches to a different shader program. context.switchShader = function(shader, posMatrix, exMatrix) { if (!posMatrix) { console.trace('posMatrix does not have required argument'); } if (this.currentShader !== shader) { this.useProgram(shader.program); // Disable all attributes from the existing shader that aren't used in // the new shader. Note: attribute indices are *not* program specific! var enabled = this.currentShader ? this.currentShader.attributes : []; var required = shader.attributes; for (var i = 0; i < enabled.length; i++) { if (required.indexOf(enabled[i]) < 0) { this.disableVertexAttribArray(enabled[i]); } } // Enable all attributes for the new shader. for (var j = 0; j < required.length; j++) { if (enabled.indexOf(required[j]) < 0) { this.enableVertexAttribArray(required[j]); } } this.currentShader = shader; } // Update the matrices if necessary. Note: This relies on object identity! // This means changing the matrix values without the actual matrix object // will FAIL to update the matrix properly. if (shader.posMatrix !== posMatrix) { this.uniformMatrix4fv(shader.u_posmatrix, false, posMatrix); shader.posMatrix = posMatrix; } if (exMatrix && shader.exMatrix !== exMatrix && shader.u_exmatrix) { this.uniformMatrix4fv(shader.u_exmatrix, false, exMatrix); shader.exMatrix = exMatrix; } }; return context; }; },{"./shaders.js":35}],33:[function(require,module,exports){ 'use strict'; var glutil = require('./glutil.js'); var browser = require('../util/browser.js'); var GlyphAtlas = require('../symbol/glyphatlas.js'); var glmatrix = require('../lib/glmatrix.js'); var FrameHistory = require('./framehistory.js'); var mat4 = glmatrix.mat4; var drawSymbol = require('./drawsymbol.js'); var drawLine = require('./drawline.js'); var drawFill = require('./drawfill.js'); var drawRaster = require('./drawraster.js'); var drawDebug = require('./drawdebug.js'); var drawBackground = require('./drawbackground.js'); var drawComposited = require('./drawcomposited.js'); var drawVertices = require('./drawvertices.js'); /* * Initialize a new painter object. * * @param {Canvas} gl an experimental-webgl drawing context */ module.exports = GLPainter; function GLPainter(gl, transform) { this.gl = glutil.extend(gl); this.transform = transform; this.framebufferObject = null; this.renderTextures = []; this.namedRenderTextures = {}; this.reusableTextures = {}; this.preFbos = {}; this.tileExtent = 4096; this.frameHistory = new FrameHistory(); this.setup(); } /* * Update the GL viewport, projection matrix, and transforms to compensate * for a new width and height value. */ GLPainter.prototype.resize = function(width, height) { var gl = this.gl; // Initialize projection matrix this.projectionMatrix = mat4.create(); mat4.ortho(this.projectionMatrix, 0, width, height, 0, 0, -1); this.width = width * browser.devicePixelRatio; this.height = height * browser.devicePixelRatio; gl.viewport(0, 0, this.width, this.height); for (var i = this.renderTextures.length - 1; i >= 0; i--) { gl.deleteTexture(this.renderTextures.pop()); } if (this.stencilBuffer) { gl.deleteRenderbuffer(this.stencilBuffer); delete this.stencilBuffer; } }; GLPainter.prototype.setup = function() { var gl = this.gl; gl.verbose = true; // We are blending the new pixels *behind* the existing pixels. That way we can // draw front-to-back and use then stencil buffer to cull opaque pixels early. gl.enable(gl.BLEND); gl.blendFunc(gl.ONE_MINUS_DST_ALPHA, gl.ONE); gl.enable(gl.STENCIL_TEST); this.glyphAtlas = new GlyphAtlas(1024, 1024); // this.glyphAtlas.debug = true; this.glyphAtlas.bind(gl); // Initialize shaders this.debugShader = gl.initializeShader('debug', ['a_pos'], ['u_posmatrix', 'u_pointsize', 'u_color']); this.compositeShader = gl.initializeShader('composite', ['a_pos'], ['u_posmatrix', 'u_opacity']); this.gaussianShader = gl.initializeShader('gaussian', ['a_pos'], ['u_posmatrix', 'u_opacity', 'u_image', 'u_offset']); this.rasterShader = gl.initializeShader('raster', ['a_pos', 'a_texture_pos'], ['u_posmatrix', 'u_brightness_low', 'u_brightness_high', 'u_saturation_factor', 'u_spin_weights', 'u_contrast_factor', 'u_opacity0', 'u_opacity1', 'u_image0', 'u_image1', 'u_tl_parent', 'u_scale_parent', 'u_buffer_scale']); this.lineShader = gl.initializeShader('line', ['a_pos', 'a_extrude', 'a_linesofar'], ['u_posmatrix', 'u_exmatrix', 'u_linewidth', 'u_color', 'u_debug', 'u_ratio', 'u_dasharray', 'u_blur']); this.linepatternShader = gl.initializeShader('linepattern', ['a_pos', 'a_extrude', 'a_linesofar'], ['u_posmatrix', 'u_exmatrix', 'u_linewidth', 'u_ratio', 'u_pattern_size', 'u_pattern_tl', 'u_pattern_br', 'u_point', 'u_blur', 'u_fade']); this.dotShader = gl.initializeShader('dot', ['a_pos'], ['u_posmatrix', 'u_size', 'u_color', 'u_blur']); this.sdfShader = gl.initializeShader('sdf', ['a_pos', 'a_tex', 'a_offset', 'a_angle', 'a_minzoom', 'a_maxzoom', 'a_rangeend', 'a_rangestart', 'a_labelminzoom'], ['u_posmatrix', 'u_exmatrix', 'u_texture', 'u_texsize', 'u_color', 'u_gamma', 'u_buffer', 'u_angle', 'u_zoom', 'u_flip', 'u_fadedist', 'u_minfadezoom', 'u_maxfadezoom', 'u_fadezoom']); this.iconShader = gl.initializeShader('icon', ['a_pos', 'a_tex', 'a_offset', 'a_angle', 'a_minzoom', 'a_maxzoom', 'a_rangeend', 'a_rangestart', 'a_labelminzoom'], ['u_posmatrix', 'u_exmatrix', 'u_texture', 'u_texsize', 'u_angle', 'u_zoom', 'u_flip', 'u_fadedist', 'u_minfadezoom', 'u_maxfadezoom', 'u_fadezoom', 'u_opacity']); this.outlineShader = gl.initializeShader('outline', ['a_pos'], ['u_posmatrix', 'u_color', 'u_world'] ); this.patternShader = gl.initializeShader('pattern', ['a_pos'], ['u_posmatrix', 'u_pattern_tl', 'u_pattern_br', 'u_mix', 'u_patternmatrix', 'u_opacity', 'u_image'] ); this.fillShader = gl.initializeShader('fill', ['a_pos'], ['u_posmatrix', 'u_color'] ); this.identityMatrix = mat4.create(); // The backgroundBuffer is used when drawing to the full *canvas* this.backgroundBuffer = gl.createBuffer(); this.backgroundBuffer.itemSize = 2; this.backgroundBuffer.itemCount = 4; gl.bindBuffer(gl.ARRAY_BUFFER, this.backgroundBuffer); gl.bufferData(gl.ARRAY_BUFFER, new Int16Array([-1, -1, 1, -1, -1, 1, 1, 1]), gl.STATIC_DRAW); // The tileExtentBuffer is used when drawing to a full *tile* this.tileExtentBuffer = gl.createBuffer(); this.tileExtentBuffer.itemSize = 4; this.tileExtentBuffer.itemCount = 4; gl.bindBuffer(gl.ARRAY_BUFFER, this.tileExtentBuffer); gl.bufferData(gl.ARRAY_BUFFER, new Int16Array([ // tile coord x, tile coord y, texture coord x, texture coord y 0, 0, 0, 0, this.tileExtent, 0, 32767, 0, 0, this.tileExtent, 0, 32767, this.tileExtent, this.tileExtent, 32767, 32767 ]), gl.STATIC_DRAW); // The debugBuffer is used to draw tile outlines for debugging this.debugBuffer = gl.createBuffer(); this.debugBuffer.itemSize = 2; this.debugBuffer.itemCount = 5; gl.bindBuffer(gl.ARRAY_BUFFER, this.debugBuffer); gl.bufferData(gl.ARRAY_BUFFER, new Int16Array([0, 0, 4095, 0, 4095, 4095, 0, 4095, 0, 0]), gl.STATIC_DRAW); // The debugTextBuffer is used to draw tile IDs for debugging this.debugTextBuffer = gl.createBuffer(); this.debugTextBuffer.itemSize = 2; }; /* * Reset the color buffers of the drawing canvas. */ GLPainter.prototype.clearColor = function() { var gl = this.gl; gl.clearColor(0, 0, 0, 0); gl.clear(gl.COLOR_BUFFER_BIT); }; /* * Reset the drawing canvas by clearing the stencil buffer so that we can draw * new tiles at the same location, while retaining previously drawn pixels. */ GLPainter.prototype.clearStencil = function() { var gl = this.gl; gl.clearStencil(0x0); gl.stencilMask(0xFF); gl.clear(gl.STENCIL_BUFFER_BIT); }; GLPainter.prototype.drawClippingMask = function() { var gl = this.gl; gl.switchShader(this.fillShader, this.tile.posMatrix, this.tile.exMatrix); gl.colorMask(false, false, false, false); // Clear the entire stencil buffer, except for the 7th bit, which stores // the global clipping mask that allows us to avoid drawing in regions of // tiles we've already painted in. gl.clearStencil(0x0); gl.stencilMask(0xBF); gl.clear(gl.STENCIL_BUFFER_BIT); // The stencil test will fail always, meaning we set all pixels covered // by this geometry to 0x80. We use the highest bit 0x80 to mark the regions // we want to draw in. All pixels that have this bit *not* set will never be // drawn in. gl.stencilFunc(gl.EQUAL, 0xC0, 0x40); gl.stencilMask(0xC0); gl.stencilOp(gl.REPLACE, gl.KEEP, gl.KEEP); // Draw the clipping mask gl.bindBuffer(gl.ARRAY_BUFFER, this.tileExtentBuffer); gl.vertexAttribPointer(this.fillShader.a_pos, this.tileExtentBuffer.itemSize, gl.SHORT, false, 8, 0); gl.drawArrays(gl.TRIANGLE_STRIP, 0, this.tileExtentBuffer.itemCount); gl.stencilFunc(gl.EQUAL, 0x80, 0x80); gl.stencilOp(gl.KEEP, gl.KEEP, gl.REPLACE); gl.stencilMask(0x00); gl.colorMask(true, true, true, true); }; // Set up a texture that can be drawn into GLPainter.prototype.bindRenderTexture = function(name) { var gl = this.gl; if (name) { // Only create one framebuffer. Reuse it for every level. if (!this.framebufferObject) { this.framebufferObject = gl.createFramebuffer(); } gl.bindFramebuffer(gl.FRAMEBUFFER, this.framebufferObject); // There's only one stencil buffer that we always attach. if (!this.stencilBuffer) { var stencil = this.stencilBuffer = gl.createRenderbuffer(); gl.bindRenderbuffer(gl.RENDERBUFFER, stencil); gl.renderbufferStorage(gl.RENDERBUFFER, gl.STENCIL_INDEX8, gl.drawingBufferWidth, gl.drawingBufferHeight); gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.STENCIL_ATTACHMENT, gl.RENDERBUFFER, this.stencilBuffer); } // We create a separate texture for every level. var texture = this.renderTextures.pop(); if (!texture) { texture = gl.createTexture(); gl.bindTexture(gl.TEXTURE_2D, texture); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST); gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.drawingBufferWidth, gl.drawingBufferHeight, 0, gl.RGBA, gl.UNSIGNED_BYTE, null); } this.namedRenderTextures[name] = texture; gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, texture, 0); } else { gl.bindFramebuffer(gl.FRAMEBUFFER, null); } this.clearColor(); }; GLPainter.prototype.freeRenderTexture = function(name) { this.renderTextures.push(this.namedRenderTextures[name]); delete this.namedRenderTextures[name]; }; /* * Draw a new tile to the context, assuming that the viewport is * already correctly set. */ GLPainter.prototype.draw = function glPainterDraw(tile, style, layers, params, matrix) { this.tile = tile; // false when drawing a group of composited layers if (tile && !matrix) { // Draw the root clipping mask. this.drawClippingMask(); } if (!Array.isArray(layers)) console.warn('Layers is not an array'); this.frameHistory.record(this.transform.zoom); // Draw layers front-to-back. // Layers are already in reverse order from style.restructure() for (var i = 0, len = layers.length; i < len; i++) { this.applyStyle(layers[i], style, tile && tile.buckets, params, tile, matrix); } if (params.debug) { drawDebug(this.gl, this, tile, params); } }; GLPainter.prototype.applyStyle = function(layer, style, buckets, params, tile, matrix) { var gl = this.gl; var layerStyle = style.computed[layer.id]; if (!layerStyle || layerStyle.hidden) return; if (layer.layers) { if (layer.type === 'composite') { drawComposited(gl, this, buckets, layerStyle, params, style, layer); } else if (layer.type === 'raster') { drawRaster(gl, this, buckets[layer.bucket], layerStyle, params, style, layer, tile); } } else if (params.background) { drawBackground(gl, this, undefined, layerStyle, this.identityMatrix, params, style.sprite); } else { var bucket = buckets[layer.bucket]; // There are no vertices yet for this layer. if (!bucket || (bucket.hasData && !bucket.hasData())) return; var type = bucket.type; if (bucket.minZoom && this.transform.zoom < bucket.minZoom) return; if (bucket.maxZoom && this.transform.zoom >= bucket.maxZoom) return; var draw = type === 'symbol' ? drawSymbol : type === 'fill' ? drawFill : type === 'line' ? drawLine : type === 'raster' ? drawRaster : null; if (draw) { var useMatrix = matrix || this.tile.posMatrix; draw(gl, this, bucket, layerStyle, useMatrix, params, style.sprite); } else { console.warn('No bucket type specified'); } if (params.vertices && !layer.layers) { drawVertices(gl, this, bucket); } } }; // Draws non-opaque areas. This is for debugging purposes. GLPainter.prototype.drawStencilBuffer = function() { var gl = this.gl; gl.switchShader(this.fillShader, this.identityMatrix); // Blend to the front, not the back. gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA); gl.stencilMask(0x00); gl.stencilFunc(gl.EQUAL, 0x80, 0x80); // Drw the filling quad where the stencil buffer isn't set. gl.bindBuffer(gl.ARRAY_BUFFER, this.backgroundBuffer); gl.vertexAttribPointer(this.fillShader.a_pos, this.backgroundBuffer.itemSize, gl.SHORT, false, 0, 0); gl.uniform4fv(this.fillShader.u_color, [0, 0, 0, 0.5]); gl.drawArrays(gl.TRIANGLE_STRIP, 0, this.backgroundBuffer.itemCount); // Revert blending mode to blend to the back. gl.blendFunc(gl.ONE_MINUS_DST_ALPHA, gl.ONE); }; GLPainter.prototype.translateMatrix = function(matrix, z, translate, anchor) { if (!translate[0] && !translate[1]) return matrix; if (anchor === 'viewport') { var sin_a = Math.sin(-this.transform.angle); var cos_a = Math.cos(-this.transform.angle); translate = [ translate[0] * cos_a - translate[1] * sin_a, translate[0] * sin_a + translate[1] * cos_a ]; } var tilePixelRatio = this.transform.scale / (1 << z) / 8; var translation = [ translate[0] / tilePixelRatio, translate[1] / tilePixelRatio, 0 ]; var translatedMatrix = new Float32Array(16); mat4.translate(translatedMatrix, matrix, translation); return translatedMatrix; }; GLPainter.prototype.saveTexture = function(texture) { var textures = this.reusableTextures[texture.size]; if (!textures) { this.reusableTextures[texture.size] = [texture]; } else { textures.push(texture); } }; GLPainter.prototype.getTexture = function(size) { var textures = this.reusableTextures[size]; return textures && textures.length > 0 ? textures.pop() : null; }; },{"../lib/glmatrix.js":21,"../symbol/glyphatlas.js":60,"../util/browser.js":77,"./drawbackground.js":23,"./drawcomposited.js":24,"./drawdebug.js":25,"./drawfill.js":26,"./drawline.js":27,"./drawraster.js":28,"./drawsymbol.js":29,"./drawvertices.js":30,"./framehistory.js":31,"./glutil.js":32}],34:[function(require,module,exports){ 'use strict'; var glmatrix = require('../lib/glmatrix.js'); var mat4 = glmatrix.mat4; module.exports = PrerenderedTexture; function PrerenderedTexture(gl, bucket, painter) { this.gl = gl; this.buffer = bucket['raster-buffer'] || (1/32); this.size = (bucket['raster-size'] || 512) * (1 + 2 * this.buffer); this.painter = painter; this.texture = null; this.fbo = null; this.fbos = this.painter.preFbos[this.size]; } PrerenderedTexture.prototype.bindFramebuffer = function() { var gl = this.gl; // try to reuse available raster textures this.texture = this.painter.getTexture(this.size); if (!this.texture) { this.texture = gl.createTexture(); gl.bindTexture(gl.TEXTURE_2D, this.texture); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR); gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, this.size, this.size, 0, gl.RGBA, gl.UNSIGNED_BYTE, null); this.texture.size = this.size; } else { gl.bindTexture(gl.TEXTURE_2D, this.texture); } if (!this.fbos) { this.fbo = gl.createFramebuffer(); var stencil = gl.createRenderbuffer(); gl.bindRenderbuffer(gl.RENDERBUFFER, stencil); gl.renderbufferStorage(gl.RENDERBUFFER, gl.STENCIL_INDEX8, this.size, this.size); gl.bindFramebuffer(gl.FRAMEBUFFER, this.fbo); gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.STENCIL_ATTACHMENT, gl.RENDERBUFFER, stencil); gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, this.texture, 0); } else { this.fbo = this.fbos.pop(); gl.bindFramebuffer(gl.FRAMEBUFFER, this.fbo); gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, this.texture, 0); } }; PrerenderedTexture.prototype.unbindFramebuffer = function() { var gl = this.gl; gl.bindFramebuffer(gl.FRAMEBUFFER, null); if (this.fbos) this.fbos.push(this.fbo); else this.painter.preFbos[this.size] = [this.fbo]; }; PrerenderedTexture.prototype.bind = function() { if (!this.texture) throw('pre-rendered texture does not exist'); var gl = this.gl; gl.bindTexture(gl.TEXTURE_2D, this.texture); }; PrerenderedTexture.prototype.blur = function(painter, passes) { var gl = this.gl; var originalTexture = this.texture; var secondaryTexture = this.painter.getTexture(this.size); if (!secondaryTexture) { secondaryTexture = gl.createTexture(); gl.bindTexture(gl.TEXTURE_2D, secondaryTexture); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR); gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, this.size, this.size, 0, gl.RGBA, gl.UNSIGNED_BYTE, null); secondaryTexture.size = this.size; } else { gl.bindTexture(gl.TEXTURE_2D, secondaryTexture); } gl.bindTexture(gl.TEXTURE_2D, null); var matrix = mat4.create(); mat4.ortho(matrix, 0, 4096, -4096, 0, 0, 1); mat4.translate(matrix, matrix, [0, -4096, 0]); gl.switchShader(painter.gaussianShader, matrix); gl.activeTexture(gl.TEXTURE0); gl.uniform1i(painter.gaussianShader.u_image, 0); gl.uniform1f(painter.gaussianShader.u_opacity, 1); for (var i = 0; i < passes; i++) { // Render horizontal gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, secondaryTexture, 0); gl.clear(gl.COLOR_BUFFER_BIT); gl.uniform2fv(painter.gaussianShader.u_offset, [1 / this.size, 0]); gl.bindTexture(gl.TEXTURE_2D, originalTexture); gl.bindBuffer(gl.ARRAY_BUFFER, painter.tileExtentBuffer); gl.vertexAttribPointer(painter.gaussianShader.a_pos, 2, gl.SHORT, false, 8, 0); gl.drawArrays(gl.TRIANGLE_STRIP, 0, 4); // Render vertical gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, originalTexture, 0); gl.clear(gl.COLOR_BUFFER_BIT); gl.uniform2fv(painter.gaussianShader.u_offset, [0, 1 / this.size]); gl.bindTexture(gl.TEXTURE_2D, secondaryTexture); gl.bindBuffer(gl.ARRAY_BUFFER, painter.tileExtentBuffer); gl.vertexAttribPointer(painter.gaussianShader.a_pos, 2, gl.SHORT, false, 8, 0); gl.drawArrays(gl.TRIANGLE_STRIP, 0, 4); } this.painter.saveTexture(secondaryTexture); }; },{"../lib/glmatrix.js":21}],35:[function(require,module,exports){ 'use strict'; var glify = undefined; module.exports = { "composite": {"vertex":"precision mediump float;\nattribute vec2 a_pos;\nuniform mat4 u_posmatrix;\nvarying highp vec2 a;\nvoid main ()\n{\n vec4 tmpvar_1;\n tmpvar_1.zw = vec2(0.0, 1.0);\n tmpvar_1.xy = a_pos;\n vec4 tmpvar_2;\n tmpvar_2 = (u_posmatrix * tmpvar_1);\n gl_Position = tmpvar_2;\n a = ((tmpvar_2.xy / 2.0) + 0.5);\n}\n\n","fragment":"precision mediump float;\nuniform sampler2D u_image;\nuniform float u_opacity;\nvarying vec2 a;\nvoid main ()\n{\n lowp vec4 tmpvar_1;\n tmpvar_1 = (texture2D (u_image, a) * u_opacity);\n gl_FragColor = tmpvar_1;\n}\n\n"}, "debug": {"vertex":"precision mediump float;\nattribute vec2 a_pos;\nuniform float u_pointsize;\nuniform mat4 u_posmatrix;\nvoid main ()\n{\n vec4 tmpvar_1;\n tmpvar_1.w = 1.0;\n tmpvar_1.xy = a_pos;\n tmpvar_1.z = float((a_pos.x >= 32767.0));\n gl_Position = (u_posmatrix * tmpvar_1);\n gl_PointSize = u_pointsize;\n}\n\n","fragment":"precision mediump float;\nuniform vec4 u_color;\nvoid main ()\n{\n gl_FragColor = u_color;\n}\n\n"}, "dot": {"vertex":"precision mediump float;\nuniform mat4 u_posmatrix;\nuniform float u_size;\nattribute vec2 a_pos;\nvoid main ()\n{\n vec4 tmpvar_1;\n tmpvar_1.zw = vec2(0.0, 1.0);\n tmpvar_1.xy = a_pos;\n gl_Position = (u_posmatrix * tmpvar_1);\n gl_PointSize = u_size;\n}\n\n","fragment":"precision mediump float;\nuniform vec4 u_color;\nuniform float u_blur;\nvoid main ()\n{\n mediump vec2 x_1;\n x_1 = (gl_PointCoord - 0.5);\n mediump float tmpvar_2;\n tmpvar_2 = clamp (((\n sqrt(dot (x_1, x_1))\n - 0.5) / (\n (0.5 - u_blur)\n - 0.5)), 0.0, 1.0);\n gl_FragColor = (u_color * (tmpvar_2 * (tmpvar_2 * \n (3.0 - (2.0 * tmpvar_2))\n )));\n}\n\n"}, "fill": {"vertex":"precision mediump float;\nattribute vec2 a_pos;\nuniform mat4 u_posmatrix;\nvoid main ()\n{\n vec4 tmpvar_1;\n tmpvar_1.zw = vec2(0.0, 1.0);\n tmpvar_1.xy = a_pos;\n gl_Position = (u_posmatrix * tmpvar_1);\n gl_PointSize = 2.0;\n}\n\n","fragment":"precision mediump float;\nuniform vec4 u_color;\nvoid main ()\n{\n gl_FragColor = u_color;\n}\n\n"}, "gaussian": {"vertex":"precision mediump float;\nattribute vec2 a_pos;\nuniform mat4 u_posmatrix;\nuniform vec2 u_offset;\nvarying highp vec2 b[3];\nvoid main ()\n{\n vec4 tmpvar_1;\n tmpvar_1.zw = vec2(0.0, 1.0);\n tmpvar_1.xy = a_pos;\n vec4 tmpvar_2;\n tmpvar_2 = (u_posmatrix * tmpvar_1);\n gl_Position = tmpvar_2;\n highp vec2 tmpvar_3;\n tmpvar_3 = ((tmpvar_2.xy / 2.0) + 0.5);\n b[0] = tmpvar_3;\n vec2 cse_4;\n cse_4 = (u_offset * 1.18243);\n b[1] = (tmpvar_3 + cse_4);\n b[2] = (tmpvar_3 - cse_4);\n}\n\n","fragment":"precision mediump float;\nuniform sampler2D u_image;\nvarying vec2 b[3];\nvoid main ()\n{\n lowp vec4 tmpvar_1;\n tmpvar_1 = (((texture2D (u_image, b[0]) * 0.40262) + (texture2D (u_image, b[1]) * 0.29869)) + (texture2D (u_image, b[2]) * 0.29869));\n gl_FragColor = tmpvar_1;\n}\n\n"}, "line": {"vertex":"precision mediump float;\nattribute vec2 a_pos;\nattribute vec2 a_extrude;\nattribute float a_linesofar;\nuniform mat4 u_posmatrix;\nuniform mat4 u_exmatrix;\nuniform float u_ratio;\nuniform vec2 u_linewidth;\nvarying vec2 a;\nvarying float b;\nvoid main ()\n{\n vec2 c_1;\n vec2 tmpvar_2;\n tmpvar_2 = (vec2(mod (a_pos, 2.0)));\n c_1.x = tmpvar_2.x;\n c_1.y = sign((tmpvar_2.y - 0.5));\n a = c_1;\n vec4 tmpvar_3;\n tmpvar_3.zw = vec2(0.0, 1.0);\n tmpvar_3.xy = floor((a_pos / 2.0));\n vec4 tmpvar_4;\n tmpvar_4.zw = vec2(0.0, 0.0);\n tmpvar_4.xy = (u_linewidth.x * (a_extrude / 63.0));\n gl_Position = ((u_posmatrix * tmpvar_3) + (u_exmatrix * tmpvar_4));\n b = (a_linesofar * u_ratio);\n}\n\n","fragment":"precision mediump float;\nuniform float u_debug;\nuniform float u_blur;\nuniform vec2 u_linewidth;\nuniform vec2 u_dasharray;\nuniform vec4 u_color;\nvarying vec2 a;\nvarying float b;\nvoid main ()\n{\n float tmpvar_1;\n tmpvar_1 = (sqrt(dot (a, a)) * u_linewidth.x);\n float tmpvar_2;\n tmpvar_2 = (float(mod (b, (u_dasharray.x + u_dasharray.y))));\n gl_FragColor = (u_color * (clamp (\n (min ((tmpvar_1 - (u_linewidth.y - u_blur)), (u_linewidth.x - tmpvar_1)) / u_blur)\n , 0.0, 1.0) * max (\n float((-(u_dasharray.y) >= 0.0))\n , \n clamp (min (tmpvar_2, (u_dasharray.x - tmpvar_2)), 0.0, 1.0)\n )));\n if ((u_debug > 0.0)) {\n gl_FragColor = vec4(1.0, 0.0, 0.0, 1.0);\n };\n}\n\n"}, "linepattern": {"vertex":"precision mediump float;\nattribute vec2 a_pos;\nattribute vec2 a_extrude;\nattribute float a_linesofar;\nuniform mat4 u_posmatrix;\nuniform mat4 u_exmatrix;\nuniform float u_point;\nuniform vec2 u_linewidth;\nvarying vec2 a;\nvarying float b;\nvoid main ()\n{\n vec2 c_1;\n vec2 tmpvar_2;\n tmpvar_2 = (vec2(mod (a_pos, 2.0)));\n c_1.x = tmpvar_2.x;\n c_1.y = sign((tmpvar_2.y - 0.5));\n a = c_1;\n vec4 tmpvar_3;\n tmpvar_3.zw = vec2(0.0, 1.0);\n tmpvar_3.xy = floor((a_pos / 2.0));\n vec4 tmpvar_4;\n tmpvar_4.w = 0.0;\n tmpvar_4.xy = ((u_linewidth.x * (a_extrude / 63.0)) * (1.0 - u_point));\n tmpvar_4.z = (float((a_pos.x >= 32767.0)) + (u_point * float(\n (c_1.y >= 1.0)\n )));\n gl_Position = ((u_posmatrix * tmpvar_3) + (u_exmatrix * tmpvar_4));\n b = a_linesofar;\n gl_PointSize = ((2.0 * u_linewidth.x) - 1.0);\n}\n\n","fragment":"precision mediump float;\nuniform float u_point;\nuniform float u_blur;\nuniform float u_fade;\nuniform vec2 u_linewidth;\nuniform vec2 u_pattern_size;\nuniform vec2 u_pattern_tl;\nuniform vec2 u_pattern_br;\nuniform sampler2D u_image;\nvarying vec2 a;\nvarying float b;\nvoid main ()\n{\n lowp vec4 j_1;\n mediump vec2 x_2;\n x_2 = ((gl_PointCoord * 2.0) - 1.0);\n mediump float tmpvar_3;\n tmpvar_3 = (((\n sqrt(dot (a, a))\n * \n (1.0 - u_point)\n ) + (u_point * \n sqrt(dot (x_2, x_2))\n )) * u_linewidth.x);\n float tmpvar_4;\n tmpvar_4 = (float(mod ((b / u_pattern_size.x), 1.0)));\n float tmpvar_5;\n tmpvar_5 = (0.5 + ((a.y * u_linewidth.x) / u_pattern_size.y));\n vec2 tmpvar_6;\n tmpvar_6.x = tmpvar_4;\n tmpvar_6.y = tmpvar_5;\n vec2 tmpvar_7;\n tmpvar_7.x = (float(mod ((tmpvar_4 * 2.0), 1.0)));\n tmpvar_7.y = tmpvar_5;\n lowp vec4 tmpvar_8;\n tmpvar_8 = ((texture2D (u_image, mix (u_pattern_tl, u_pattern_br, tmpvar_6)) * (1.0 - u_fade)) + (u_fade * texture2D (u_image, mix (u_pattern_tl, u_pattern_br, tmpvar_7))));\n j_1.w = tmpvar_8.w;\n j_1.xyz = (tmpvar_8.xyz * tmpvar_8.w);\n gl_FragColor = (j_1 * clamp ((\n min ((tmpvar_3 - (u_linewidth.y - u_blur)), (u_linewidth.x - tmpvar_3))\n / u_blur), 0.0, 1.0));\n}\n\n"}, "outline": {"vertex":"precision mediump float;\nattribute vec2 a_pos;\nuniform mat4 u_posmatrix;\nuniform vec2 u_world;\nvarying highp vec2 a;\nvoid main ()\n{\n vec4 tmpvar_1;\n tmpvar_1.zw = vec2(0.0, 1.0);\n tmpvar_1.xy = a_pos;\n vec4 tmpvar_2;\n tmpvar_2 = (u_posmatrix * tmpvar_1);\n gl_Position = tmpvar_2;\n a = (((tmpvar_2.xy + 1.0) / 2.0) * u_world);\n}\n\n","fragment":"precision mediump float;\nuniform vec4 u_color;\nvarying vec2 a;\nvoid main ()\n{\n highp vec2 x_1;\n x_1 = (a - gl_FragCoord.xy);\n highp float tmpvar_2;\n tmpvar_2 = clamp (((\n sqrt(dot (x_1, x_1))\n - 1.0) / -1.0), 0.0, 1.0);\n highp vec4 tmpvar_3;\n tmpvar_3 = (u_color * (tmpvar_2 * (tmpvar_2 * \n (3.0 - (2.0 * tmpvar_2))\n )));\n gl_FragColor = tmpvar_3;\n}\n\n"}, "pattern": {"vertex":"precision mediump float;\nuniform mat4 u_posmatrix;\nuniform mat3 u_patternmatrix;\nattribute vec2 a_pos;\nvarying vec2 a;\nvoid main ()\n{\n vec4 tmpvar_1;\n tmpvar_1.zw = vec2(0.0, 1.0);\n tmpvar_1.xy = a_pos;\n gl_Position = (u_posmatrix * tmpvar_1);\n vec3 tmpvar_2;\n tmpvar_2.z = 1.0;\n tmpvar_2.xy = a_pos;\n a = (u_patternmatrix * tmpvar_2).xy;\n}\n\n","fragment":"precision mediump float;\nuniform float u_opacity;\nuniform float u_mix;\nuniform vec2 u_pattern_tl;\nuniform vec2 u_pattern_br;\nuniform sampler2D u_image;\nvarying vec2 a;\nvoid main ()\n{\n vec2 tmpvar_1;\n tmpvar_1 = (vec2(mod (a, 1.0)));\n lowp vec4 tmpvar_2;\n tmpvar_2 = (mix (texture2D (u_image, mix (u_pattern_tl, u_pattern_br, tmpvar_1)), texture2D (u_image, mix (u_pattern_tl, u_pattern_br, \n (vec2(mod ((tmpvar_1 * 2.0), 1.0)))\n )), u_mix) * u_opacity);\n gl_FragColor = tmpvar_2;\n}\n\n"}, "raster": {"vertex":"precision mediump float;\nuniform mat4 u_posmatrix;\nuniform vec2 u_tl_parent;\nuniform float u_scale_parent;\nuniform float u_buffer_scale;\nattribute vec2 a_pos;\nattribute vec2 a_texture_pos;\nvarying vec2 a;\nvarying vec2 b;\nvoid main ()\n{\n vec4 tmpvar_1;\n tmpvar_1.zw = vec2(0.0, 1.0);\n tmpvar_1.xy = a_pos;\n gl_Position = (u_posmatrix * tmpvar_1);\n vec2 tmpvar_2;\n tmpvar_2 = (((\n (a_texture_pos / 32767.0)\n - 0.5) / u_buffer_scale) + 0.5);\n a = tmpvar_2;\n b = ((tmpvar_2 * u_scale_parent) + u_tl_parent);\n}\n\n","fragment":"precision mediump float;\nuniform float u_opacity0;\nuniform float u_opacity1;\nuniform float u_brightness_low;\nuniform float u_brightness_high;\nuniform float u_saturation_factor;\nuniform float u_contrast_factor;\nuniform sampler2D u_image0;\nuniform sampler2D u_image1;\nvarying vec2 a;\nvarying vec2 b;\nuniform vec3 u_spin_weights;\nvoid main ()\n{\n lowp vec4 tmpvar_1;\n tmpvar_1 = ((texture2D (u_image0, a) * u_opacity0) + (texture2D (u_image1, b) * u_opacity1));\n lowp vec3 tmpvar_2;\n tmpvar_2.x = dot (tmpvar_1.xyz, u_spin_weights);\n tmpvar_2.y = dot (tmpvar_1.xyz, u_spin_weights.zxy);\n tmpvar_2.z = dot (tmpvar_1.xyz, u_spin_weights.yzx);\n lowp vec4 tmpvar_3;\n tmpvar_3.xyz = mix (vec3(u_brightness_low), vec3(u_brightness_high), ((\n ((tmpvar_2 + ((\n (((tmpvar_1.x + tmpvar_1.y) + tmpvar_1.z) / 3.0)\n - tmpvar_2) * u_saturation_factor)) - 0.5)\n * u_contrast_factor) + 0.5));\n tmpvar_3.w = tmpvar_1.w;\n gl_FragColor = tmpvar_3;\n}\n\n"}, "icon": {"vertex":"precision mediump float;\nattribute vec2 a_pos;\nattribute vec2 a_offset;\nattribute vec2 a_tex;\nattribute float a_angle;\nattribute float a_minzoom;\nattribute float a_maxzoom;\nattribute float a_rangeend;\nattribute float a_rangestart;\nattribute float a_labelminzoom;\nuniform mat4 u_posmatrix;\nuniform mat4 u_exmatrix;\nuniform float u_angle;\nuniform float u_zoom;\nuniform float u_flip;\nuniform float u_fadedist;\nuniform float u_minfadezoom;\nuniform float u_maxfadezoom;\nuniform float u_fadezoom;\nuniform float u_opacity;\nuniform vec2 u_texsize;\nvarying vec2 a;\nvarying float b;\nvoid main ()\n{\n float d_1;\n d_1 = 0.0;\n float tmpvar_2;\n tmpvar_2 = (float(mod ((a_angle + u_angle), 256.0)));\n if ((((u_flip > 0.0) && (tmpvar_2 >= 64.0)) && (tmpvar_2 < 192.0))) {\n d_1 = 1.0;\n };\n float tmpvar_3;\n tmpvar_3 = (((2.0 - \n float((u_zoom >= a_minzoom))\n ) - (1.0 - \n float((u_zoom >= a_maxzoom))\n )) + d_1);\n float tmpvar_4;\n tmpvar_4 = clamp (((u_fadezoom - a_labelminzoom) / u_fadedist), 0.0, 1.0);\n if ((u_fadedist >= 0.0)) {\n b = tmpvar_4;\n } else {\n b = (1.0 - tmpvar_4);\n };\n if ((u_maxfadezoom < a_labelminzoom)) {\n b = 0.0;\n };\n if ((u_minfadezoom >= a_labelminzoom)) {\n b = 1.0;\n };\n vec4 tmpvar_5;\n tmpvar_5.zw = vec2(0.0, 1.0);\n tmpvar_5.xy = a_pos;\n vec4 tmpvar_6;\n tmpvar_6.w = 0.0;\n tmpvar_6.xy = (a_offset / 64.0);\n tmpvar_6.z = ((tmpvar_3 + float(\n (0.0 >= b)\n )) + (float(\n (u_angle >= a_rangeend)\n ) * (1.0 - \n float((u_angle >= a_rangestart))\n )));\n gl_Position = ((u_posmatrix * tmpvar_5) + (u_exmatrix * tmpvar_6));\n a = (a_tex / u_texsize);\n b = (b * u_opacity);\n}\n\n","fragment":"precision mediump float;\nuniform sampler2D u_texture;\nvarying vec2 a;\nvarying float b;\nvoid main ()\n{\n lowp vec4 tmpvar_1;\n tmpvar_1 = (texture2D (u_texture, a) * b);\n gl_FragColor = tmpvar_1;\n}\n\n"}, "sdf": {"vertex":"precision mediump float;\nattribute vec2 a_pos;\nattribute vec2 a_offset;\nattribute vec2 a_tex;\nattribute float a_angle;\nattribute float a_minzoom;\nattribute float a_maxzoom;\nattribute float a_rangeend;\nattribute float a_rangestart;\nattribute float a_labelminzoom;\nuniform mat4 u_posmatrix;\nuniform mat4 u_exmatrix;\nuniform float u_angle;\nuniform float u_zoom;\nuniform float u_flip;\nuniform float u_fadedist;\nuniform float u_minfadezoom;\nuniform float u_maxfadezoom;\nuniform float u_fadezoom;\nuniform vec2 u_texsize;\nvarying vec2 a;\nvarying float b;\nvoid main ()\n{\n float d_1;\n d_1 = 0.0;\n float tmpvar_2;\n tmpvar_2 = (float(mod ((a_angle + u_angle), 256.0)));\n if ((((u_flip > 0.0) && (tmpvar_2 >= 64.0)) && (tmpvar_2 < 192.0))) {\n d_1 = 1.0;\n };\n float tmpvar_3;\n tmpvar_3 = (((2.0 - \n float((u_zoom >= a_minzoom))\n ) - (1.0 - \n float((u_zoom >= a_maxzoom))\n )) + d_1);\n float tmpvar_4;\n tmpvar_4 = clamp (((u_fadezoom - a_labelminzoom) / u_fadedist), 0.0, 1.0);\n if ((u_fadedist >= 0.0)) {\n b = tmpvar_4;\n } else {\n b = (1.0 - tmpvar_4);\n };\n if ((u_maxfadezoom < a_labelminzoom)) {\n b = 0.0;\n };\n if ((u_minfadezoom >= a_labelminzoom)) {\n b = 1.0;\n };\n vec4 tmpvar_5;\n tmpvar_5.zw = vec2(0.0, 1.0);\n tmpvar_5.xy = a_pos;\n vec4 tmpvar_6;\n tmpvar_6.w = 0.0;\n tmpvar_6.xy = (a_offset / 64.0);\n tmpvar_6.z = ((tmpvar_3 + float(\n (0.0 >= b)\n )) + (float(\n (u_angle >= a_rangeend)\n ) * (1.0 - \n float((u_angle >= a_rangestart))\n )));\n gl_Position = ((u_posmatrix * tmpvar_5) + (u_exmatrix * tmpvar_6));\n a = (a_tex / u_texsize);\n}\n\n","fragment":"precision mediump float;\nuniform sampler2D u_texture;\nuniform vec4 u_color;\nuniform float u_buffer;\nuniform float u_gamma;\nvarying vec2 a;\nvarying float b;\nvoid main ()\n{\n float edge0_1;\n edge0_1 = (u_buffer - u_gamma);\n lowp float tmpvar_2;\n tmpvar_2 = clamp (((texture2D (u_texture, a).w - edge0_1) / (\n (u_buffer + u_gamma)\n - edge0_1)), 0.0, 1.0);\n lowp vec4 tmpvar_3;\n tmpvar_3 = (u_color * ((tmpvar_2 * \n (tmpvar_2 * (3.0 - (2.0 * tmpvar_2)))\n ) * b));\n gl_FragColor = tmpvar_3;\n}\n\n"} }; },{}],36:[function(require,module,exports){ 'use strict'; var Source = require('./source.js'); var GeoJSONTile = require('./geojsontile.js'); var GeoJSONSource = module.exports = function(options) { this.tiles = {}; this.alltiles = {}; this.enabled = true; this.zooms = [1, 5, 9, 13]; this.minTileZoom = this.zooms[0]; this.maxTileZoom = this.zooms[this.zooms.length - 1]; this.loadNewTiles = true; this.tileJSON = { minZoom: 1, maxZoom: 13 }; this.data = options.data; }; GeoJSONSource.prototype = Object.create(Source.prototype); GeoJSONSource.prototype.setData = function(data) { this.data = data; if (this.map) this._updateData(); return this; }; GeoJSONSource.prototype.onAdd = function(map) { this.map = map; this.painter = map.painter; if (this.map.style) this._updateData(); map.on('style.change', this._updateData.bind(this)); }; GeoJSONSource.prototype._updateData = function() { var source = this; this.workerID = this.map.dispatcher.send('parse geojson', { data: this.data, zooms: this.zooms, tileSize: 512, source: this.id }, function(err, tiles) { if (err) return; for (var i = 0; i < tiles.length; i++) { source.alltiles[tiles[i].id] = new GeoJSONTile(tiles[i].id, source, tiles[i]); } if (source.map) source.map.update(); }); return this; }; GeoJSONSource.prototype._addTile = function(id) { var tile = this.alltiles[id]; if (tile) { tile._load(); this.tiles[id] = tile; this.fire('tile.add', {tile: tile}); } return tile || {}; }; GeoJSONSource.prototype._coveringZoomLevel = function(zoom) { for (var i = this.zooms.length - 1; i >= 0; i--) { if (this.zooms[i] <= zoom) { var z = this.zooms[i]; return z; } } return 0; }; },{"./geojsontile.js":37,"./source.js":40}],37:[function(require,module,exports){ 'use strict'; var Tile = require('./tile.js'); var BufferSet = require('../data/buffer/bufferset.js'); var createBucket = require('../data/createbucket.js'); module.exports = GeoJSONTile; function GeoJSONTile(id, source, data) { this.id = id; this.source = source; this.data = data; this.workerID = source.workerID; } GeoJSONTile.prototype = Object.create(Tile.prototype); GeoJSONTile.prototype._load = function() { if (this.loaded) return; this.loaded = true; var data = this.data; this.buffers = new BufferSet(data.buffers); this.buckets = {}; for (var b in data.elementGroups) { this.buckets[b] = createBucket(this.source.map.style.buckets[b], this.buffers, undefined, data.elementGroups[b]); } }; // noops GeoJSONTile.prototype.abort = function() { }; GeoJSONTile.prototype.remove = function() { }; },{"../data/buffer/bufferset.js":2,"../data/createbucket.js":10,"./tile.js":41}],38:[function(require,module,exports){ 'use strict'; module.exports = Wrapper; // conform to vectortile api function Wrapper(features) { this.features = features; this.length = features.length; } Wrapper.prototype.feature = function(i) { return new FeatureWrapper(this.features[i]); }; var mapping = { 'Point': 1, 'LineString': 2, 'Polygon': 3 }; function FeatureWrapper(feature) { this.feature = feature; this._type = mapping[feature.type]; this.properties = feature.properties; } FeatureWrapper.prototype.loadGeometry = function() { return this.feature.coords; }; FeatureWrapper.prototype.bbox = function() { if (this._type === mapping.Point) { return [ this.feature.coords[0], this.feature.coords[1], this.feature.coords[0], this.feature.coords[1] ]; } var rings = this.feature.coords; var x1 = Infinity, x2 = -Infinity, y1 = Infinity, y2 = -Infinity; for (var i = 0; i < rings.length; i++) { var ring = rings[i]; for (var j = 0; j < ring.length; j++) { var coord = ring[j]; x1 = Math.min(x1, coord.x); x2 = Math.max(x2, coord.x); y1 = Math.min(y1, coord.y); y2 = Math.max(y2, coord.y); } } return [x1, y1, x2, y2]; }; },{}],39:[function(require,module,exports){ 'use strict'; var Tile = require('./tile.js'); var ajax = require('../util/ajax.js'); module.exports = RasterTile; function RasterTile(id, source, url, callback) { this.id = id; this.loaded = false; this.url = url; this.source = source; this.map = source.map; this._load(); this.callback = callback; this.uses = 1; // Todo finish figuring out what raster buckets are this.buckets = {}; this.info = { raster: true }; var buckets = this.map.style.buckets; for (var b in buckets) { var bucket = buckets[b]; var sourceid = bucket && bucket.source; if (source.id === sourceid) { this.buckets[b] = { info: bucket.render, type: 'raster', tile: this }; } } } RasterTile.prototype = Object.create(Tile.prototype); RasterTile.prototype._load = function() { var tile = this; ajax.getImage(this.url, function(err, img) { // @TODO handle errors. if (err) return; tile.img = img; if (tile.map) tile.onTileLoad(); }); }; RasterTile.prototype.onTileLoad = function() { // start texture upload this.bind(this.map.painter.gl); this.loaded = true; this.callback(); }; RasterTile.prototype.abort = function() { this.aborted = true; if (this.img) this.img.src = ''; delete this.img; }; RasterTile.prototype.bind = function(gl) { if (!this.texture) { // try to find reusable texture this.texture = this.map.painter.getTexture(this.img.width); if (this.texture) { gl.bindTexture(gl.TEXTURE_2D, this.texture); gl.texSubImage2D(gl.TEXTURE_2D, 0, 0, 0, gl.RGBA, gl.UNSIGNED_BYTE, this.img); } else { this.texture = gl.createTexture(); gl.bindTexture(gl.TEXTURE_2D, this.texture); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR_MIPMAP_NEAREST); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, this.img); this.texture.size = this.img.width; } gl.generateMipmap(gl.TEXTURE_2D); } else { gl.bindTexture(gl.TEXTURE_2D, this.texture); } }; RasterTile.prototype.remove = function() { if (this.texture) this.map.painter.saveTexture(this.texture); delete this.map; }; RasterTile.prototype.featuresAt = function(pos, params, callback) { // noop callback(null, []); }; },{"../util/ajax.js":76,"./tile.js":41}],40:[function(require,module,exports){ 'use strict'; var util = require('../util/util.js'), ajax = require('../util/ajax.js'), tileJSON = require('../util/url.js').tileJSON, Evented = require('../util/evented.js'), Cache = require('../util/mrucache.js'), TileCoord = require('./tilecoord'), VectorTile = require('./vectortile'), RasterTile = require('./rastertile.js'), Point = require('point-geometry'); module.exports = Source; Source.protocols = { "mapbox": function(url, callback) { ajax.getJSON(tileJSON(url.split('://')[1]), callback); } }; function Source(options) { this.tiles = {}; this.enabled = false; this.type = options.type; if (this.type === 'vector' && options.tileSize && options.tileSize !== 512) { throw new Error('vector tile sources must have a tileSize of 512'); } this.Tile = this.type === 'vector' ? VectorTile : RasterTile; this.options = util.inherit(this.options, options); this.cache = new Cache(this.options.cacheSize, function(tile) { tile.remove(); }); var protocol = options.url.split(':')[0]; Source.protocols[protocol](options.url, function(err, tileJSON) { if (err) throw err; this.tileJSON = tileJSON; this.loadNewTiles = true; this.enabled = true; this.update(); if (this.map) this.map.fire('source.add', {source: this}); }.bind(this)); this._updateTiles = util.throttle(this._updateTiles, 50, this); } Source.prototype = util.inherit(Evented, { options: { tileSize: 512, cacheSize: 20 }, onAdd: function(map) { this.map = map; this.painter = map.painter; }, load: function() { for (var t in this.tiles) { this.tiles[t]._load(); } }, loaded: function() { for (var t in this.tiles) { if (!this.tiles[t].loaded) return false; } return true; }, update: function() { if (!this.enabled) return; this._updateTiles(); }, render: function(layers) { // Iteratively paint every tile. if (!this.enabled) return; var order = Object.keys(this.tiles); order.sort(this._z_order); for (var i = 0; i < order.length; i++) { var id = order[i]; var tile = this.tiles[id]; if (tile.loaded && !this.coveredTiles[id]) { this._renderTile(tile, id, layers); } } }, featuresAt: function(point, params, callback) { point = Point.convert(point); if (params.layer) { var style = this.map.style, layer = style.getLayer(params.layer); params.bucket = style.buckets[layer.ref || layer.id]; } var order = Object.keys(this.tiles); order.sort(this._z_order); for (var i = 0; i < order.length; i++) { var id = order[i]; var tile = this.tiles[id]; var pos = tile.positionAt(id, point); if (pos && pos.x >= 0 && pos.x < 4096 && pos.y >= 0 && pos.y < 4096) { // The click is within the viewport. There is only ever one tile in // a layer that has this property. return tile.featuresAt(pos, params, callback); } } callback(null, []); }, // get the zoom level adjusted for the difference in map and source tilesizes _getZoom: function() { var zOffset = Math.log(this.map.transform.tileSize/this.options.tileSize) / Math.LN2; return this.map.transform.zoom + zOffset; }, _coveringZoomLevel: function(zoom) { for (var z = this.tileJSON.maxzoom; z >= this.tileJSON.minzoom; z--) { if (z <= zoom) { if (this.type === 'raster') { // allow underscaling by rounding to the nearest zoom level if (z < this.tileJSON.maxzoom) { z += Math.round(zoom - z); } } return z; } } return 0; }, _childZoomLevel: function(zoom) { zoom = Math.max(this.tileJSON.minzoom, zoom + 1); return zoom <= this.tileJSON.maxzoom ? zoom : null; }, _getCoveringTiles: function(zoom) { if (zoom === undefined) zoom = this._getZoom(); var z = this._coveringZoomLevel(zoom), tiles = 1 << z, tr = this.map.transform, tileCenter = TileCoord.zoomTo(tr.locationCoordinate(tr.center), z); var points = [ TileCoord.zoomTo(tr.pointCoordinate(tileCenter, {x: 0, y: 0}), z), TileCoord.zoomTo(tr.pointCoordinate(tileCenter, {x: tr.width, y: 0}), z), TileCoord.zoomTo(tr.pointCoordinate(tileCenter, {x: tr.width, y: tr.height}), z), TileCoord.zoomTo(tr.pointCoordinate(tileCenter, {x: 0, y: tr.height}), z) ], t = {}; // Divide the screen up in two triangles and scan each of them: // +---/ // | / | // /---+ this._scanTriangle(points[0], points[1], points[2], 0, tiles, scanLine); this._scanTriangle(points[2], points[3], points[0], 0, tiles, scanLine); return Object.keys(t).sort(fromCenter); function fromCenter(a, b) { var ad = Math.abs(a.x - tileCenter.column) + Math.abs(a.y - tileCenter.row), bd = Math.abs(b.x - tileCenter.column) + Math.abs(b.y - tileCenter.row); return ad - bd; } function scanLine(x0, x1, y) { var x, wx; if (y >= 0 && y <= tiles) { for (x = x0; x < x1; x++) { wx = (x + tiles) % tiles; t[TileCoord.toID(z, wx, y, Math.floor(x/tiles))] = {x: wx, y: y}; } } } }, // Given a tile of data, its id, and a style layers, render the tile to the canvas _renderTile: function(tile, id, layers) { var pos = TileCoord.fromID(id); var z = pos.z, x = pos.x, y = pos.y, w = pos.w; x += w * (1 << z); tile.calculateMatrices(z, x, y, this.map.transform, this.painter); this.painter.draw(tile, this.map.style, layers, { z: z, x: x, y: y, debug: this.map.debug, antialiasing: this.map.antialiasing, vertices: this.map.vertices, rotating: this.map.rotating, zooming: this.map.zooming }); }, // Recursively find children of the given tile (up to maxCoveringZoom) that are already loaded; // adds found tiles to retain object; returns true if children completely cover the tile _findLoadedChildren: function(id, maxCoveringZoom, retain) { var complete = true; var z = TileCoord.fromID(id).z; var ids = TileCoord.children(id); for (var i = 0; i < ids.length; i++) { if (this.tiles[ids[i]] && this.tiles[ids[i]].loaded) { retain[ids[i]] = true; } else { complete = false; if (z < maxCoveringZoom) { // Go further down the hierarchy to find more unloaded children. this._findLoadedChildren(ids[i], maxCoveringZoom, retain); } } } return complete; }, // Find a loaded parent of the given tile (up to minCoveringZoom); // adds the found tile to retain object and returns true if a parent was found _findLoadedParent: function(id, minCoveringZoom, retain) { for (var z = TileCoord.fromID(id).z; z >= minCoveringZoom; z--) { id = TileCoord.parent(id); if (this.tiles[id] && this.tiles[id].loaded) { retain[id] = true; return true; } } return false; }, // Removes tiles that are outside the viewport and adds new tiles that are inside the viewport. _updateTiles: function() { if (!this.map || !this.map.loadNewTiles || !this.loadNewTiles || !this.map.style.sources[this.id]) return; var zoom = Math.floor(this._getZoom()); var required = this._getCoveringTiles().sort(this._centerOut.bind(this)); var i; var id; var complete; var tile; // Determine the overzooming/underzooming amounts. var minCoveringZoom = Math.max(this.tileJSON.minzoom, zoom - 10); var maxCoveringZoom = this.tileJSON.minzoom; while (maxCoveringZoom < zoom + 1) { var level = this._childZoomLevel(maxCoveringZoom); if (level === null) break; else maxCoveringZoom = level; } // Retain is a list of tiles that we shouldn't delete, even if they are not // the most ideal tile for the current viewport. This may include tiles like // parent or child tiles that are *already* loaded. var retain = {}; // Covered is a list of retained tiles who's areas are full covered by other, // better, retained tiles. They are not drawn separately. this.coveredTiles = {}; var fullyComplete = true; // Add existing child/parent tiles if the actual tile is not yet loaded for (i = 0; i < required.length; i++) { id = +required[i]; retain[id] = true; tile = this._addTile(id); if (!tile.loaded) { // The tile we require is not yet loaded. Try to find a parent or // child tile that we already have. // First, try to find existing child tiles that completely cover the // missing tile. complete = this._findLoadedChildren(id, maxCoveringZoom, retain); // Then, if there are no complete child tiles, try to find existing // parent tiles that completely cover the missing tile. if (!complete) { complete = this._findLoadedParent(id, minCoveringZoom, retain); } // The unloaded tile's area is not completely covered loaded tiles if (!complete) { fullyComplete = false; } } } var now = new Date().getTime(); var fadeDuration = this.type === 'raster' ? this.map.style.rasterFadeDuration : 0; for (id in retain) { tile = this.tiles[id]; if (tile && tile.timeAdded > now - fadeDuration) { // This tile is still fading in. Find tiles to cross-fade with it. complete = this._findLoadedChildren(id, maxCoveringZoom, retain); if (complete) { this.coveredTiles[id] = true; } else { this._findLoadedParent(id, minCoveringZoom, retain); } } } for (id in this.coveredTiles) retain[id] = true; // Remove the tiles we don't need anymore. var remove = util.keysDifference(this.tiles, retain); for (i = 0; i < remove.length; i++) { id = +remove[i]; this._removeTile(id); } }, _loadTile: function(id) { var layer = this; var map = this.map, pos = TileCoord.fromID(id), tile; if (pos.w === 0) { // console.time('loading ' + pos.z + '/' + pos.x + '/' + pos.y); var url = TileCoord.url(id, this.tileJSON.tiles); tile = this.tiles[id] = new this.Tile(id, this, url, tileComplete); } else { var wrapped = TileCoord.toID(pos.z, pos.x, pos.y, 0); tile = this.tiles[id] = this.tiles[wrapped] || this._addTile(wrapped); tile.uses++; } function tileComplete(err) { // console.timeEnd('loading ' + pos.z + '/' + pos.x + '/' + pos.y); if (err) { console.warn('failed to load tile %d/%d/%d: %s', pos.z, pos.x, pos.y, err.stack || err); } else { layer.fire('tile.load', {tile: tile}); map.update(); } } return tile; }, // Adds a vector tile to the map. It will trigger a rerender of the map and will // be part in all future renders of the map. The map object will handle copying // the tile data to the GPU if it is required to paint the current viewport. _addTile: function(id) { var tile = this.tiles[id]; if (!tile) { tile = this.cache.get(id); if (tile) { tile.uses = 1; this.tiles[id] = tile; } } if (!tile) { tile = this._loadTile(id); this.fire('tile.add', {tile: tile}); } if (tile && tile.loaded && !tile.timeAdded) { tile.timeAdded = new Date().getTime(); if (this.type === 'raster') { this.map.animationLoop.set(this.map.style.rasterFadeDuration); } } return tile; }, _removeTile: function(id) { var tile = this.tiles[id]; if (tile) { tile.uses--; delete this.tiles[id]; if (tile.uses <= 0) { delete tile.timeAdded; if (!tile.loaded) { tile.abort(); tile.remove(); } else { this.cache.add(id, tile); } this.fire('tile.remove', {tile: tile}); } } }, // Taken from polymaps src/Layer.js // https://github.com/simplegeo/polymaps/blob/master/src/Layer.js#L333-L383 // scan-line conversion _scanTriangle: function(a, b, c, ymin, ymax, scanLine) { var ab = this._edge(a, b), bc = this._edge(b, c), ca = this._edge(c, a); var t; // sort edges by y-length if (ab.dy > bc.dy) { t = ab; ab = bc; bc = t; } if (ab.dy > ca.dy) { t = ab; ab = ca; ca = t; } if (bc.dy > ca.dy) { t = bc; bc = ca; ca = t; } // scan span! scan span! if (ab.dy) this._scanSpans(ca, ab, ymin, ymax, scanLine); if (bc.dy) this._scanSpans(ca, bc, ymin, ymax, scanLine); }, // scan-line conversion _edge: function(a, b) { if (a.row > b.row) { var t = a; a = b; b = t; } return { x0: a.column, y0: a.row, x1: b.column, y1: b.row, dx: b.column - a.column, dy: b.row - a.row }; }, // scan-line conversion _scanSpans: function(e0, e1, ymin, ymax, scanLine) { var y0 = Math.max(ymin, Math.floor(e1.y0)), y1 = Math.min(ymax, Math.ceil(e1.y1)); // sort edges by x-coordinate if ((e0.x0 == e1.x0 && e0.y0 == e1.y0) ? (e0.x0 + e1.dy / e0.dy * e0.dx < e1.x1) : (e0.x1 - e1.dy / e0.dy * e0.dx < e1.x0)) { var t = e0; e0 = e1; e1 = t; } // scan lines! var m0 = e0.dx / e0.dy, m1 = e1.dx / e1.dy, d0 = e0.dx > 0, // use y + 1 to compute x0 d1 = e1.dx < 0; // use y + 1 to compute x1 for (var y = y0; y < y1; y++) { var x0 = m0 * Math.max(0, Math.min(e0.dy, y + d0 - e0.y0)) + e0.x0, x1 = m1 * Math.max(0, Math.min(e1.dy, y + d1 - e1.y0)) + e1.x0; scanLine(Math.floor(x1), Math.ceil(x0), y); } }, _z_order: function(a, b) { return (b % 32) - (a % 32); }, _centerOut: function(a, b) { var tr = this.map.transform; var aPos = TileCoord.fromID(a); var bPos = TileCoord.fromID(b); var c = TileCoord.zoomTo(tr.locationCoordinate(tr.center), aPos.z); var center = new Point(c.column - 0.5, c.row - 0.5); return center.dist(aPos) - center.dist(bPos); }, }); var sources = { vector: Source, raster: Source, geojson: require('./geojsonsource'), video: require('./videosource') }; Source.create = function(source) { return new sources[source.type](source); }; },{"../util/ajax.js":76,"../util/evented.js":82,"../util/mrucache.js":84,"../util/url.js":86,"../util/util.js":87,"./geojsonsource":36,"./rastertile.js":39,"./tilecoord":42,"./vectortile":44,"./videosource":45,"point-geometry":98}],41:[function(require,module,exports){ 'use strict'; var glmatrix = require('../lib/glmatrix.js'), mat2 = glmatrix.mat2, mat4 = glmatrix.mat4, vec2 = glmatrix.vec2; module.exports = Tile; function Tile() {} Tile.prototype = { // todo unhardcode tileExtent: 4096, calculateMatrices: function(z, x, y, transform, painter) { // Initialize model-view matrix that converts from the tile coordinates // to screen coordinates. var tileScale = Math.pow(2, z); var scale = transform.worldSize / tileScale; // TODO: remove this.scale = scale; // The position matrix this.posMatrix = mat4.create(); mat4.translate(this.posMatrix, this.posMatrix, [transform.centerPoint.x, transform.centerPoint.y, 0]); mat4.rotateZ(this.posMatrix, this.posMatrix, transform.angle); mat4.translate(this.posMatrix, this.posMatrix, [-transform.centerPoint.x, -transform.centerPoint.y, 0]); var pixelX = transform.width / 2 - transform.x, pixelY = transform.height / 2 - transform.y; mat4.translate(this.posMatrix, this.posMatrix, [pixelX + x * scale, pixelY + y * scale, 1]); // Create inverted matrix for interaction this.invPosMatrix = mat4.create(); mat4.invert(this.invPosMatrix, this.posMatrix); mat4.scale(this.posMatrix, this.posMatrix, [ scale / this.tileExtent, scale / this.tileExtent, 1 ]); mat4.multiply(this.posMatrix, painter.projectionMatrix, this.posMatrix); // The extrusion matrix. this.exMatrix = mat4.clone(painter.projectionMatrix); mat4.rotateZ(this.exMatrix, this.exMatrix, transform.angle); // 2x2 matrix for rotating points this.rotationMatrix = mat2.create(); mat2.rotate(this.rotationMatrix, this.rotationMatrix, transform.angle); }, positionAt: function(id, point) { // tile hasn't finished loading if (!this.invPosMatrix) return null; var pos = vec2.transformMat4([], [point.x, point.y], this.invPosMatrix); vec2.scale(pos, pos, 4096 / this.scale); return { x: pos[0], y: pos[1], scale: this.scale }; }, featuresAt: function(pos, params, callback) { this.source.map.dispatcher.send('query features', { id: this.id, x: pos.x, y: pos.y, scale: pos.scale, source: this.source.id, params: params }, callback, this.workerID); } }; },{"../lib/glmatrix.js":21}],42:[function(require,module,exports){ 'use strict'; /* * Tiles are generally represented as packed integer ids constructed by * `TileCoord.toID(x, y, z)` */ var TileCoord = exports; TileCoord.toID = function(z, x, y, w) { w = w || 0; w *= 2; if (w < 0) w = w * -1 -1; var dim = 1 << z; return ((dim * dim * w + dim * y + x) * 32) + z; }; TileCoord.asString = function(id) { var pos = TileCoord.fromID(id); return pos.z + "/" + pos.x + "/" + pos.y; }; /* * Parse a packed integer id into an object with x, y, and z properties */ TileCoord.fromID = function(id) { var z = id % 32, dim = 1 << z; var xy = ((id - z) / 32); var x = xy % dim, y = ((xy - x) / dim) % dim; var w = Math.floor(xy / (dim * dim)); if (w % 2 !== 0) w = w * -1 -1; w /= 2; return { z: z, x: x, y: y, w: w }; }; /* * Given a packed integer id, return its zoom level */ TileCoord.zoom = function(id) { return id % 32; }; // Given an id and a list of urls, choose a url template and return a tile URL TileCoord.url = function(id, urls) { var pos = TileCoord.fromID(id); return urls[Math.floor((pos.x + pos.y) % urls.length)] .replace('{h}', (pos.x % 16).toString(16) + (pos.y % 16).toString(16)) .replace('{z}', pos.z.toFixed(0)) .replace('{x}', pos.x.toFixed(0)) .replace('{y}', pos.y.toFixed(0)); }; /* * Given a packed integer id, return the id of its parent tile */ TileCoord.parent = function(id) { var pos = TileCoord.fromID(id); if (pos.z === 0) return id; else return TileCoord.toID(pos.z - 1, Math.floor(pos.x / 2), Math.floor(pos.y / 2)); }; TileCoord.parentWithZoom = function(id, zoom) { var pos = TileCoord.fromID(id); while (pos.z > zoom) { pos.z--; pos.x = Math.floor(pos.x / 2); pos.y = Math.floor(pos.y / 2); } return TileCoord.toID(pos.z, pos.x, pos.y); }; /* * Given a packed integer id, return an array of integer ids representing * its four children. */ TileCoord.children = function(id) { var pos = TileCoord.fromID(id); pos.z += 1; pos.x *= 2; pos.y *= 2; return [ TileCoord.toID(pos.z, pos.x, pos.y, pos.w), TileCoord.toID(pos.z, pos.x + 1, pos.y, pos.w), TileCoord.toID(pos.z, pos.x, pos.y + 1, pos.w), TileCoord.toID(pos.z, pos.x + 1, pos.y + 1, pos.w) ]; }; TileCoord.zoomTo = function(c, z) { c.column = c.column * Math.pow(2, z - c.zoom); c.row = c.row * Math.pow(2, z - c.zoom); c.zoom = z; return c; }; },{}],43:[function(require,module,exports){ 'use strict'; var rewind = require('geojson-rewind'); var TileCoord = require('./tilecoord.js'); var Transform = require('../geo/transform.js'); var Point = require('point-geometry'); var LatLng = require('../geo/latlng.js'); module.exports = tileGeoJSON; function tileGeoJSON(geojson, zoom) { var tiles = {}; var tileExtent = 4096; var transform = new Transform(); transform.zoom = zoom; geojson = rewind(geojson); if (geojson.type === 'FeatureCollection') { for (var i = 0; i < geojson.features.length; i++) { tileFeature(geojson.features[i], transform, tiles, tileExtent); } } else if (geojson.type === 'Feature') { tileFeature(geojson, transform, tiles, tileExtent); } else { throw('Unrecognized geojson type'); } return tiles; } function tileFeature(feature, transform, tiles, tileExtent) { var coords = feature.geometry.coordinates; var type = feature.geometry.type; var tiled; if (type === 'Point') { tiled = tileLineString([coords], transform, tileExtent); } else if (type === 'LineString' || type === 'MultiPoint') { tiled = tileLineString(coords, transform, tileExtent); } else if (type === 'Polygon' || type === 'MultiLineString') { tiled = {}; for (var i = 0; i < coords.length; i++) { var tiled_ = tileLineString(coords[i], transform, tileExtent, type === 'Polygon'); for (var tileID in tiled_) { if (!tiled[tileID]) tiled[tileID] = []; tiled[tileID] = (tiled[tileID] || []).concat(tiled_[tileID]); } } } else if (type === 'MultiPolygon') { throw("todo"); } else { throw("unrecognized geometry type"); } for (var id in tiled) { tiles[id] = tiles[id] || []; tiles[id].push({ properties: feature.properties, coords: tiled[id], type: feature.geometry.type }); } } function tileLineString(coords, transform, tileExtent, rejoin) { var padding = 0.01; var paddedExtent = tileExtent * (1 + 2 * padding); var coord = transform.locationCoordinate(new LatLng(coords[0][1], coords[0][0])); var prevCoord; var tiles = {}; for (var i = 0; i < coords.length; i++) { prevCoord = coord; coord = transform.locationCoordinate(new LatLng(coords[i][1], coords[i][0])); var dx = coord.column - prevCoord.column || Number.MIN_VALUE, dy = coord.row - prevCoord.row || Number.MIN_VALUE, dirX = dx / Math.abs(dx), dirY = dy / Math.abs(dy); // Find the rectangular bounding box, in tiles, of the polygon var startTileX = Math.floor(prevCoord.column - dirX * padding); var endTileX = Math.floor(coord.column + dirX * padding); var startTileY = Math.floor(prevCoord.row - dirY * padding); var endTileY = Math.floor(coord.row + dirY * padding); // Iterate over all tiles the segment might intersect // and split the segment across those tiles for (var x = startTileX; (x - endTileX) * dirX <= 0; x += dirX) { var leftX = (x - padding - prevCoord.column) / dx; var rightX = (x + 1 + padding - prevCoord.column) / dx; for (var y = startTileY; (y - endTileY) * dirY <= 0; y += dirY) { var topY = (y - padding - prevCoord.row) / dy; var bottomY = (y + 1 + padding - prevCoord.row) / dy; // fraction of the distance along the segment at which the segment // enters or exits the tile var enter = Math.max(Math.min(leftX, rightX), Math.min(topY, bottomY)); var exit = Math.min(Math.max(leftX, rightX), Math.max(topY, bottomY)); var tileID = TileCoord.toID(transform.tileZoom, x, y), tile = tiles[tileID], point; // segments starts outside the tile, add entry point if (0 <= enter && enter < 1) { point = new Point( ((prevCoord.column + enter * dx) - x) * tileExtent, ((prevCoord.row + enter * dy) - y) * tileExtent); point.continues = true; if (!tile) tiles[tileID] = tile = []; tile.push([point]); } // segments ends outside the tile, add exit point if (0 <= exit && exit < 1) { point = new Point( ((prevCoord.column + exit * dx) - x) * tileExtent, ((prevCoord.row + exit * dy) - y) * tileExtent); point.continues = true; tile[tile.length - 1].push(point); // add the point itself } else { point = new Point( (coord.column - x) * tileExtent, (coord.row - y) * tileExtent); if (!tile) tiles[tileID] = tile = [[point]]; else tile[tile.length - 1].push(point); } } } } if (rejoin) { // reassemble the disconnected segments into a linestring // sections of the linestring outside the tile are replaced with segments // that follow the tile's edge for (var id in tiles) { var segments = tiles[id]; if (!segments[0][0].continues && segments.length > 1) { // if the first segment is the beginning of the linestring // then join it with the last so that all segments start and // end at tile boundaries var last = segments.pop(); Array.prototype.unshift.apply(segments[0], last.slice(0, last.length - 1)); } var start = edgeDist(segments[0][0], tileExtent, padding); for (var k = 0; k < segments.length; k++) { // Add all tile corners along the path between the current segment's exit point // and the next segment's entry point var thisExit = edgeDist(segments[k][segments[k].length - 1], paddedExtent); var nextEntry = edgeDist(segments[(k + 1) % segments.length][0], paddedExtent); var startToExit = (thisExit - start + 4) % 4; var startToNextEntry = (nextEntry - start + 4) % 4; var direction = (thisExit === nextEntry || startToExit < startToNextEntry) ? 1 : -1; var roundFn = direction > 0 ? Math.ceil : Math.floor; for (var c = roundFn(thisExit) % 4; c != roundFn(nextEntry) % 4; c = (c + direction + 4) % 4) { var corner = corners[c]; segments[k].push(new Point( (corner.x + (corner.x - 0.5 > 0 ? 1 : -1) * padding) * tileExtent, (corner.y + (corner.y - 0.5 > 0 ? 1 : -1) * padding) * tileExtent)); } } // Join all segments tiles[id] = [Array.prototype.concat.apply([], segments)]; } } return tiles; } var corners = [ new Point(0, 0), new Point(1, 0), new Point(1, 1), new Point(0, 1)]; /* * Converts to a point to the distance along the edge of the tile (out of 4). * * 0.5 * 0 _______ 1 * | | * 3.5 | | 1.5 * | | * |_______| * 3 2.5 2 */ function edgeDist(point, extent) { var x = point.x / extent; var y = point.y / extent; var d; if (Math.abs(y - 0.5) >= Math.abs(x - 0.5)) { d = Math.round(y) * 2 + (y < 0.5 ? x : 1 - x); } else { d = Math.round(1 - x) * 2 + (x > 0.5 ? y : 1 - y) + 1; } return d % 4; } },{"../geo/latlng.js":17,"../geo/transform.js":19,"./tilecoord.js":42,"geojson-rewind":92,"point-geometry":98}],44:[function(require,module,exports){ 'use strict'; var Tile = require('./tile.js'), TileCoord = require('./tilecoord.js'), BufferSet = require('../data/buffer/bufferset.js'), util = require('../util/util.js'); var createBucket = require('../data/createbucket.js'); module.exports = VectorTile; function VectorTile(id, source, url, callback) { this.id = id; this.loaded = false; this.url = url; this.zoom = TileCoord.fromID(id).z; this.map = source.map; this.options = source.options; this.id = util.uniqueId(); this.callback = callback; this.source = source; if (this.zoom >= source.tileJSON.maxzoom) { this.depth = this.map.options.maxZoom - this.zoom; } else { this.depth = 1; } this.uses = 1; this._load(); } VectorTile.prototype = util.inherit(Tile, { _load: function() { var tile = this; this.workerID = this.map.dispatcher.send('load tile', { url: this.url, id: this.id, zoom: this.zoom, maxZoom: this.source.tileJSON.maxzoom, tileSize: this.options.tileSize, source: this.source.id, depth: this.depth }, function(err, data) { if (!err && data) { tile.onTileLoad(data); } tile.callback(err); }); }, onTileLoad: function(data) { // Tile has been removed from the map if (!this.map) return; this.buffers = new BufferSet(data.buffers); this.buckets = {}; for (var b in data.elementGroups) { this.buckets[b] = createBucket(this.map.style.buckets[b], this.buffers, undefined, data.elementGroups[b]); } this.loaded = true; }, remove: function() { // reuse prerendered textures for (var bucket in this.buckets) { if (this.buckets[bucket].prerendered) this.map.painter.saveTexture(this.buckets[bucket].prerendered.texture); } this.map.dispatcher.send('remove tile', { id: this.id, source: this.source.id }, null, this.workerID); this.map.painter.glyphAtlas.removeGlyphs(this.id); var gl = this.map.painter.gl; var buffers = this.buffers; if (buffers) { for (var b in buffers) { buffers[b].destroy(gl); } } delete this.map; }, abort: function() { this.map.dispatcher.send('abort tile', { id: this.id, source: this.source.id }, null, this.workerID); } }); },{"../data/buffer/bufferset.js":2,"../data/createbucket.js":10,"../util/util.js":87,"./tile.js":41,"./tilecoord.js":42}],45:[function(require,module,exports){ 'use strict'; var Tile = require('./tile.js'); var TileCoord = require('./tilecoord.js'); var LatLng = require('../geo/latlng.js'); var Point = require('point-geometry'); module.exports = VideoSource; function VideoSource(options) { this.video = document.createElement('video'); this.video.crossOrigin = 'Anonymous'; this.video.loop = true; var urls = (typeof options.url === 'string') ? [options.url] : options.url; for (var i = 0; i < urls.length; i++) { var s = document.createElement('source'); s.src = urls[i]; this.video.appendChild(s); } this.coordinates = options.coordinates; this.enabled = true; var loopID; var source = this; // start repainting when video starts playing this.video.addEventListener('playing', function() { loopID = source.map.style.animationLoop.set(Infinity); source.map._rerender(); }); // stop repainting when video stops this.video.addEventListener('pause', function() { source.map.style.animationLoop.cancel(loopID); }); } VideoSource.prototype.onAdd = function(map) { this.map = map; this.video.play(); this.createTile(); }; VideoSource.prototype.createTile = function() { /* * Calculate which mercator tile is suitable for rendering the video in * and create a buffer with the corner coordinates. These coordinates * may be outside the tile, because raster tiles aren't clipped when rendering. */ var map = this.map; var coords = this.coordinates.map(function(latlng) { var loc = LatLng.convert(latlng); return TileCoord.zoomTo(map.transform.locationCoordinate(loc), 0); }); var minX = Infinity; var minY = Infinity; var maxX = -Infinity; var maxY = -Infinity; for (var i = 0; i < coords.length; i++) { minX = Math.min(minX, coords[i].column); minY = Math.min(minY, coords[i].row); maxX = Math.max(maxX, coords[i].column); maxY = Math.max(maxY, coords[i].row); } var dx = maxX - minX; var dy = maxY - minY; var dMax = Math.max(dx, dy); var center = TileCoord.zoomTo({ column: (minX + maxX) / 2, row: (minY + maxY) / 2, zoom: 0 }, Math.floor(-Math.log(dMax) / Math.LN2)); var tileExtent = 4096; var tileCoords = coords.map(function(coord) { var zoomedCoord = TileCoord.zoomTo(coord, center.zoom); return new Point( Math.round((zoomedCoord.column - center.column) * tileExtent), Math.round((zoomedCoord.row - center.row) * tileExtent)); }); var gl = map.painter.gl; var maxInt16 = 32767; var array = new Int16Array([ tileCoords[0].x, tileCoords[0].y, 0, 0, tileCoords[1].x, tileCoords[1].y, maxInt16, 0, tileCoords[3].x, tileCoords[3].y, 0, maxInt16, tileCoords[2].x, tileCoords[2].y, maxInt16, maxInt16 ]); this.boundsBuffer = gl.createBuffer(); gl.bindBuffer(gl.ARRAY_BUFFER, this.boundsBuffer); gl.bufferData(gl.ARRAY_BUFFER, array, gl.STATIC_DRAW); this.tile = new Tile(); this.center = center; }; VideoSource.prototype.load = function() { // noop }; VideoSource.prototype.update = function() { // noop }; VideoSource.prototype.render = function(layers) { if (!this.enabled) return; if (this.video.readyState < 2) return; // not enough data for current position var layer = layers[0]; var bucket = { type: 'raster', tile: this, boundsBuffer: this.boundsBuffer, bind: this.bind.bind(this) }; var buckets = {}; buckets[layer.bucket] = bucket; var c = this.center; this.tile.calculateMatrices(c.zoom, c.column, c.row, this.map.transform, this.map.painter); this.map.painter.tile = this.tile; this.map.painter.applyStyle(layer, this.map.style, buckets, {}); }; VideoSource.prototype.bind = function(gl) { if (!this.texture) { this.texture = gl.createTexture(); gl.bindTexture(gl.TEXTURE_2D, this.texture); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR); gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, this.video); } else { gl.bindTexture(gl.TEXTURE_2D, this.texture); gl.texSubImage2D(gl.TEXTURE_2D, 0, 0, 0, gl.RGBA, gl.UNSIGNED_BYTE, this.video); } }; VideoSource.prototype.featuresAt = function(point, params, callback) { // TODO return pixel? return callback(null, []); }; },{"../geo/latlng.js":17,"./tile.js":41,"./tilecoord.js":42,"point-geometry":98}],46:[function(require,module,exports){ 'use strict'; var Actor = require('../util/actor.js'), bucketFilter = require('../style/bucketfilter.js'), WorkerTile = require('./workertile.js'), tileGeoJSON = require('./tilegeojson.js'), Wrapper = require('./geojsonwrapper.js'), util = require('../util/util.js'), queue = require('queue-async'), ajax = require('../util/ajax.js'); module.exports = Worker; function Worker(self) { this.self = self; this.actor = new Actor(self, this); } util.extend(Worker.prototype, { alert: function() { this.self.postMessage({ type: 'alert message', data: [].slice.call(arguments) }); }, // Updates the style to use for this map. 'set buckets': function(data) { var buckets = WorkerTile.buckets = data; for (var i = 0; i < buckets.length; i++) { var bucket = buckets[i]; bucket.compare = bucketFilter(bucket.filter); } }, 'set glyphs': function(data) { WorkerTile.prototype.glyphs = data; }, /* * Load and parse a tile at `url`, and call `callback` with * (err, response) * * @param {string} url * @param {function} callback */ 'load tile': function(params, callback) { new WorkerTile(params.url, undefined, params.id, params.zoom, params.maxZoom, params.tileSize, params.source, params.depth, this.actor, callback); }, /* * Abort the request keyed under `url` * * @param {string} url */ 'abort tile': function(params) { WorkerTile.cancel(params.id, params.source); }, 'remove tile': function(params) { var id = params.id; var source = params.source; if (WorkerTile.loaded[source] && WorkerTile.loaded[source][id]) { delete WorkerTile.loaded[source][id]; } }, 'parse geojson': function(params, callback) { var data = params.data, zooms = params.zooms, len = zooms.length, maxZoom = zooms[len - 1], actor = this.actor, q = queue(); function worker(id, tile, zoom, callback) { new WorkerTile(undefined, new Wrapper(tile), id, zoom, maxZoom, params.tileSize, params.source, 4, actor, function(err, data) { if (err) return callback(err); data.id = id; callback(null, data); }); } function tileData(err, data) { if (err) throw err; for (var i = 0; i < len; i++) { var zoom = zooms[i]; var tiles = tileGeoJSON(data, zoom); for (var id in tiles) { q.defer(worker, id, tiles[id], zoom); } } q.awaitAll(callback); } if (typeof data === 'string') ajax.getJSON(data, tileData); else tileData(null, data); }, 'query features': function(params, callback) { var tile = WorkerTile.loaded[params.source] && WorkerTile.loaded[params.source][params.id]; if (tile) { tile.featureTree.query(params, callback); } else { callback(null, []); } } }); },{"../style/bucketfilter.js":49,"../util/actor.js":75,"../util/ajax.js":76,"../util/util.js":87,"./geojsonwrapper.js":38,"./tilegeojson.js":43,"./workertile.js":47,"queue-async":99}],47:[function(require,module,exports){ 'use strict'; var FeatureTree = require('../data/featuretree.js'); var Protobuf = require('pbf'); var vt = require('vector-tile'); var Collision = require('../symbol/collision.js'); var getArrayBuffer = require('../util/ajax.js').getArrayBuffer; var BufferSet = require('../data/buffer/bufferset.js'); var createBucket = require('../data/createbucket.js'); module.exports = WorkerTile; function WorkerTile(url, data, id, zoom, maxZoom, tileSize, source, depth, actor, callback) { var tile = this; this.id = id; this.zoom = zoom; this.maxZoom = maxZoom; this.tileSize = tileSize; this.source = source; this.depth = depth; this.buffers = new BufferSet(); function loaded(data) { WorkerTile.loaded[source] = WorkerTile.loaded[source] || {}; WorkerTile.loaded[source][id] = tile; tile.data = data; tile.parse(data, actor, callback); } if (url) { if (WorkerTile.loading[source] === undefined) WorkerTile.loading[source] = {}; WorkerTile.loading[source][id] = getArrayBuffer(url, function(err, data) { delete WorkerTile.loading[source][id]; if (err) { callback(err); } else { loaded(new vt.VectorTile(new Protobuf(new Uint8Array(data)))); } }); } else { loaded(data); } } WorkerTile.cancel = function(id, sourceID) { var source = WorkerTile.loading[sourceID]; if (source && source[id]) { source[id].abort(); delete source[id]; } }; // Stores tiles that are currently loading. WorkerTile.loading = {}; // Stores tiles that are currently loaded. WorkerTile.loaded = {}; // Stores the style information. WorkerTile.buckets = []; /* * Given tile data, parse raw vertices and data, create a vector * tile and parse it into ready-to-render vertices. * * @param {object} data * @param {function} respond */ WorkerTile.prototype.parse = function(data, actor, callback) { var tile = this; var bucketInfo = WorkerTile.buckets; this.callback = callback; var tileExtent = 4096; this.collision = new Collision(this.zoom, tileExtent, this.tileSize, this.depth); this.featureTree = new FeatureTree(getGeometry, getType); var buckets = this.buckets = sortTileIntoBuckets(this, data, bucketInfo); var key, bucket; var prevPlacementBucket; var remaining = WorkerTile.buckets.length; /* * The async parsing here is a bit tricky. * Some buckets depend on resources that may need to be loaded async (glyphs). * Some buckets need to be parsed in order (to get placement priorities right). * * Dependencies calls are initiated first to get those rolling. * Buckets that don't need to be parsed in order, aren't to save time. */ var orderedBuckets = WorkerTile.buckets; for (var i = 0; i < orderedBuckets.length; i++) { bucket = buckets[orderedBuckets[i].id]; if (!bucket) { remaining--; continue; // raster bucket, etc } var filter = bucket.info.filter; if (filter && filter.source !== this.source) continue; // Link buckets that need to be parsed in order if (bucket.collision) { if (prevPlacementBucket) { prevPlacementBucket.next = bucket; } else { bucket.previousPlaced = true; } prevPlacementBucket = bucket; } if (bucket.getDependencies) { bucket.getDependencies(this, actor, dependenciesDone(bucket)); } } // parse buckets where order doesn't matter and no dependencies for (key in buckets) { bucket = buckets[key]; if (!bucket.getDependencies && !bucket.collision) { parseBucket(tile, bucket); } } function dependenciesDone(bucket) { return function(err) { bucket.dependenciesLoaded = true; parseBucket(tile, bucket, err); }; } function parseBucket(tile, bucket, skip) { if (bucket.getDependencies && !bucket.dependenciesLoaded) return; if (bucket.collision && !bucket.previousPlaced) return; if (!skip) { var now = Date.now(); if (bucket.type !== 'raster') bucket.addFeatures(); var time = Date.now() - now; if (bucket.interactive) { for (var i = 0; i < bucket.features.length; i++) { var feature = bucket.features[i]; tile.featureTree.insert(feature.bbox(), bucket.name, feature); } } if (typeof self !== 'undefined') { self.bucketStats = self.bucketStats || {_total: 0}; self.bucketStats._total += time; self.bucketStats[bucket.name] = (self.bucketStats[bucket.name] || 0) + time; } } remaining--; if (!remaining) return tile.done(); // try parsing the next bucket, if it is ready if (bucket.next) { bucket.next.previousPlaced = true; parseBucket(tile, bucket.next); } } }; WorkerTile.prototype.done = function() { // Collect all buffers to mark them as transferable object. var buffers = []; for (var type in this.buffers) { buffers.push(this.buffers[type].array); } // Convert buckets to a transferable format var buckets = this.buckets; var elementGroups = {}; for (var b in buckets) elementGroups[b] = buckets[b].elementGroups; this.callback(null, { elementGroups: elementGroups, buffers: this.buffers }, buffers); // we don't need anything except featureTree at this point, so we mark it for GC this.buffers = null; this.collision = null; this.buckets = null; }; function sortTileIntoBuckets(tile, data, bucketInfo) { var sourceLayers = {}, buckets = {}, layerName; // For each source layer, find a list of buckets that use data from it for (var i = 0; i < bucketInfo.length; i++) { var info = bucketInfo[i]; var bucketName = info.id; var minZoom = info['min-zoom']; var maxZoom = info['max-zoom']; if (info.source !== tile.source) continue; if (minZoom && tile.zoom < minZoom && minZoom < tile.maxZoom) continue; if (maxZoom && tile.zoom >= maxZoom) continue; var bucket = createBucket(info, tile.buffers, tile.collision); if (!bucket) continue; bucket.features = []; bucket.name = bucketName; buckets[bucketName] = bucket; if (data.layers) { // vectortile layerName = info['source-layer']; if (!sourceLayers[layerName]) sourceLayers[layerName] = {}; sourceLayers[layerName][bucketName] = info; } else { // geojson tile sourceLayers[bucketName] = info; } } // read each layer, and sort its feature's into buckets if (data.layers) { // vectortile for (layerName in sourceLayers) { var layer = data.layers[layerName]; if (!layer) continue; sortLayerIntoBuckets(layer, sourceLayers[layerName], buckets); } } else { // geojson sortLayerIntoBuckets(data, sourceLayers, buckets); } return buckets; } /* * Sorts features in a layer into different buckets, according to the maping * * Layers in vector tiles contain many different features, and feature types, * e.g. the landuse layer has parks, industrial buildings, forests, playgrounds * etc. However, when styling, we need to separate these features so that we can * render them separately with different styles. * * @param {VectorTileLayer} layer * @param {Mapping} mapping */ function sortLayerIntoBuckets(layer, mapping, buckets) { for (var i = 0; i < layer.length; i++) { var feature = layer.feature(i); for (var key in mapping) { if (mapping[key].compare(feature)) { buckets[key].features.push(feature); } } } } function getGeometry(feature) { return feature.loadGeometry(); } function getType(feature) { return vt.VectorTileFeature.types[feature.type]; } },{"../data/buffer/bufferset.js":2,"../data/createbucket.js":10,"../data/featuretree.js":12,"../symbol/collision.js":59,"../util/ajax.js":76,"pbf":96,"vector-tile":102}],48:[function(require,module,exports){ 'use strict'; module.exports = AnimationLoop; function AnimationLoop() { this.n = 0; this.times = []; } // Are all animations done? AnimationLoop.prototype.stopped = function() { this.times = this.times.filter(function(t) { return t.time >= (new Date()).getTime(); }); return !this.times.length; }; // Add a new animation that will run t milliseconds // Returns an id that can be used to cancel it layer AnimationLoop.prototype.set = function(t) { this.times.push({ id: this.n, time: t + (new Date()).getTime() }); return this.n++; }; // Cancel an animation AnimationLoop.prototype.cancel = function(n) { this.times = this.times.filter(function(t) { return t.id != n; }); }; },{}],49:[function(require,module,exports){ 'use strict'; var VectorTileFeature = require('vector-tile').VectorTileFeature; function infix(operator) { return function(left, right) { return left + ' ' + operator + ' ' + right; }; } var infixOperators = { '==': infix('==='), '>': infix('>'), '$gt': infix('>'), '<': infix('<'), '$lt': infix('<'), '<=': infix('<='), '$lte': infix('<='), '>=': infix('>='), '$gte': infix('>='), '!=': infix('!=='), '$ne': infix('!=='), '$exists': function (value) { return value + ' !== undefined'; } }; function or(items) { return '(' + items.join(' || ') + ')'; } function and(items) { return '(' + items.join(' && ') + ')'; } function not(item) { return '!' + item; } function nor(items) { return not(or(items)); } var arrayOperators = { '||': or, '$or': or, '&&': and, '$and': and, '!': nor, '$nor': nor }; var objOperators = { '!': not, '$not': not }; module.exports = function (filter) { // simple key & value comparison function valueFilter(key, value, operator) { return operator('p[' + JSON.stringify(key) + ']', JSON.stringify(value)); } // compares key & value or key & or(values) function simpleFieldFilter(key, value, operator) { var operatorFn = infixOperators[operator || '==']; if (!operatorFn) throw new Error('Unknown operator: ' + operator); if (Array.isArray(value)) { return or(value.map(function (v) { return valueFilter(key, v, operatorFn); })); } else return valueFilter(key, value, operatorFn); } // handles any filter key/value pair function fieldFilter(key, value) { if (Array.isArray(value)) { if (key in arrayOperators) { // handle and/or operators return arrayOperators[key](value.map(fieldsFilter)); } } else if (typeof value === 'object') { // handle not operator if (key in objOperators) return objOperators[key](fieldsFilter(value)); // handle {key: {operator: value}} notation var filters = []; for (var op in value) { filters.push(simpleFieldFilter(key, value[op], op)); } return and(filters); } // handle simple key/value or key/values comparison return simpleFieldFilter(key, value); } function typeFilter(type) { return 'f.type === ' + VectorTileFeature.types.indexOf(type); } function fieldsFilter(obj) { var filters = []; for (var key in obj) { if (key === '$type') { filters.push(typeFilter(obj[key])); } else { filters.push(fieldFilter(key, obj[key])); } } return filters.length ? and(filters) : 'true'; } var filterStr = 'var p = f.properties || {}; return ' + fieldsFilter(filter || {}) + ';'; // jshint evil: true return new Function('f', filterStr); }; },{"vector-tile":102}],50:[function(require,module,exports){ 'use strict'; var reference = require('mapbox-gl-style-spec/reference/v4'); module.exports = {}; reference['class'].forEach(function(className) { var Calculated = function() {}; var style = reference[className]; for (var prop in style) { if (style[prop]['default'] === undefined) continue; Calculated.prototype[prop] = style[prop]['default']; } module.exports[className.replace('class_','')] = Calculated; }); },{"mapbox-gl-style-spec/reference/v4":95}],51:[function(require,module,exports){ 'use strict'; var Evented = require('../util/evented.js'); var ajax = require('../util/ajax.js'); var browser = require('../util/browser.js'); module.exports = ImageSprite; function ImageSprite(base) { var sprite = this; this.base = base; this.retina = browser.devicePixelRatio > 1; base = sprite.base + (sprite.retina ? '@2x' : ''); ajax.getJSON(base + '.json', function(err, data) { // @TODO handle errors via sprite event. if (err) return; sprite.data = data; if (sprite.img) sprite.fire('loaded'); }); ajax.getImage(base + '.png', function(err, img) { // @TODO handle errors via sprite event. if (err) return; // premultiply the sprite var data = img.getData(); var newdata = img.data = new Uint8Array(data.length); for (var i = 0; i < data.length; i+=4) { var alpha = data[i + 3] / 255; newdata[i + 0] = data[i + 0] * alpha; newdata[i + 1] = data[i + 1] * alpha; newdata[i + 2] = data[i + 2] * alpha; newdata[i + 3] = data[i + 3]; } sprite.img = img; if (sprite.data) sprite.fire('loaded'); }); } ImageSprite.prototype = Object.create(Evented); ImageSprite.prototype.toJSON = function() { return this.base; }; ImageSprite.prototype.loaded = function() { return !!(this.data && this.img); }; ImageSprite.prototype.resize = function(gl) { var sprite = this; if (browser.devicePixelRatio > 1 !== sprite.retina) { var newSprite = new ImageSprite(sprite.base); newSprite.on('loaded', function() { sprite.img = newSprite.img; sprite.data = newSprite.data; sprite.retina = newSprite.retina; if (sprite.texture) { gl.deleteTexture(sprite.texture); delete sprite.texture; } }); } }; ImageSprite.prototype.bind = function(gl, linear) { var sprite = this; if (!sprite.loaded()) return; if (!sprite.texture) { sprite.texture = gl.createTexture(); gl.bindTexture(gl.TEXTURE_2D, sprite.texture); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); var img = sprite.img; gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, img.width, img.height, 0, gl.RGBA, gl.UNSIGNED_BYTE, img.data); } else { gl.bindTexture(gl.TEXTURE_2D, sprite.texture); } var filter = linear ? gl.LINEAR : gl.NEAREST; if (filter !== sprite.filter) { gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, filter); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, filter); } }; ImageSprite.prototype.getPosition = function(name, repeating) { // `repeating` indicates that the image will be used in a repeating pattern // repeating pattern images are assumed to have a 1px padding that mirrors the opposite edge // positions for repeating images are adjusted to exclude the edge repeating = repeating === true ? 1 : 0; var pos = this.data && this.data[name]; if (pos && this.img) { var width = this.img.width; var height = this.img.height; return { size: [pos.width / pos.pixelRatio, pos.height / pos.pixelRatio], tl: [(pos.x + repeating)/ width, (pos.y + repeating) / height], br: [(pos.x + pos.width - 2 * repeating) / width, (pos.y + pos.height - 2 * repeating) / height] }; } }; },{"../util/ajax.js":76,"../util/browser.js":77,"../util/evented.js":82}],52:[function(require,module,exports){ 'use strict'; var reference = require('mapbox-gl-style-spec/reference/v4'); module.exports = {}; reference.render.forEach(function(className) { var Properties = function(props) { for (var p in props) { this[p] = props[p]; } }; var properties = reference[className]; for (var prop in properties) { if (properties[prop]['default'] === undefined) continue; Properties.prototype[prop] = properties[prop]['default']; } module.exports[className.replace('render_','')] = Properties; }); },{"mapbox-gl-style-spec/reference/v4":95}],53:[function(require,module,exports){ 'use strict'; var Evented = require('../util/evented.js'); var StyleTransition = require('./styletransition.js'); var StyleDeclaration = require('./styledeclaration.js'); var StyleConstant = require('./styleconstant.js'); var CalculatedStyle = require('./calculatedstyle.js'); var ImageSprite = require('./imagesprite.js'); var util = require('../util/util.js'); module.exports = Style; /* * The map style's current state * * The stylesheet object is not modified. To change the style, just change * the the stylesheet object and trigger a cascade. */ function Style(stylesheet, animationLoop) { if (stylesheet.version !== 4) console.warn('Stylesheet version must be 4'); if (!Array.isArray(stylesheet.layers)) console.warn('Stylesheet must have layers'); this.classes = {}; this.stylesheet = stylesheet; this.animationLoop = animationLoop; this.buckets = {}; this.orderedBuckets = []; this.transitions = {}; this.computed = {}; this.sources = {}; this.cascade({transition: false}); if (stylesheet.sprite) this.setSprite(stylesheet.sprite); } Style.prototype = Object.create(Evented); function premultiplyLayer(layer, type) { var colorProp = type + '-color', haloProp = type + '-halo-color', outlineProp = type + '-outline-color', color = layer[colorProp], haloColor = layer[haloProp], outlineColor = layer[outlineProp], opacity = layer[type + '-opacity']; var colorOpacity = color && (opacity * color[3]); var haloOpacity = haloColor && (opacity * haloColor[3]); var outlineOpacity = outlineColor && (opacity * outlineColor[3]); if (colorOpacity !== undefined && colorOpacity < 1) { layer[colorProp] = util.premultiply([color[0], color[1], color[2], colorOpacity]); } if (haloOpacity !== undefined && haloOpacity < 1) { layer[haloProp] = util.premultiply([haloColor[0], haloColor[1], haloColor[2], haloOpacity]); } if (outlineOpacity !== undefined && outlineOpacity < 1) { layer[outlineProp] = util.premultiply([outlineColor[0], outlineColor[1], outlineColor[2], outlineOpacity]); } } // Formerly known as zoomed styles Style.prototype.recalculate = function(z) { if (typeof z !== 'number') console.warn('recalculate expects zoom level'); var transitions = this.transitions; var layerValues = {}; this.sources = {}; this.rasterFadeDuration = 300; for (var name in transitions) { var layer = transitions[name], bucket = this.buckets[layer.ref || name], layerType = this.layermap[name].type; if (!CalculatedStyle[layerType]) { console.warn('unknown layer type ' + layerType); continue; } var appliedLayer = layerValues[name] = new CalculatedStyle[layerType](); for (var rule in layer) { var transition = layer[rule]; appliedLayer[rule] = transition.at(z); } if (layerType === 'symbol') { appliedLayer.hidden = (appliedLayer['text-opacity'] === 0 || !bucket.render['text-field']) && (appliedLayer['icon-opacity'] === 0 || !bucket.render['icon-image']); if (!appliedLayer.hidden) { premultiplyLayer(appliedLayer, 'text'); premultiplyLayer(appliedLayer, 'icon'); } } else { appliedLayer.hidden = (appliedLayer[layerType + '-opacity'] === 0); if (!appliedLayer.hidden) { premultiplyLayer(appliedLayer, layerType); } } // Find all the sources that are currently being used // so that we can automatically enable/disable them as needed if (!appliedLayer.hidden) { var source = bucket && bucket.source; // mark source as used so that tiles are downloaded if (source) this.sources[source] = true; } if (appliedLayer['raster-fade']) { this.rasterFadeDuration = Math.max(this.rasterFadeDuration, appliedLayer['raster-fade']); } } this.computed = layerValues; this.z = z; this.fire('zoom'); }; Style.prototype._simpleLayer = function(layer) { var simple = {}; simple.id = layer.id; var bucket = this.buckets[layer.ref || layer.id]; if (bucket) simple.bucket = bucket.id; if (layer.type) simple.type = layer.type; if (layer.layers) { simple.layers = []; for (var i = 0; i < layer.layers.length; i++) { simple.layers.push(this._simpleLayer(layer.layers[i])); } } return simple; }; // Split the layers into groups of consecutive layers with the same datasource // For each group calculate its dependencies. Its dependencies are composited // layers that need to be rendered into textures before Style.prototype._groupLayers = function(layers) { var g = 0; var groups = []; var group; // loop over layers top down for (var i = layers.length - 1; i >= 0; i--) { var layer = layers[i]; var bucket = this.buckets[layer.ref || layer.id]; var source = bucket && bucket.source; // if the current layer is in a different source if (group && source !== group.source) g++; if (!groups[g]) { group = []; group.source = source; if (layer.layers) group.composited = true; groups[g] = group; } if (layer.layers && layer.type == 'composite') { // TODO if composited layer is opaque just inline the layers group.dependencies = group.dependencies || {}; group.dependencies[layer.id] = this._groupLayers(layer.layers); } group.push(this._simpleLayer(layer)); } return groups; }; /* * Take all the rules and declarations from the stylesheet, * and figure out which apply currently */ Style.prototype.cascade = function(options) { options = options || { transition: true }; var a, b; var id; var prop; var layer; var className; var styleName; var style; var styleTrans; var constants = this.stylesheet.constants; // derive buckets from layers this.orderedBuckets = []; this.buckets = getbuckets({}, this.orderedBuckets, this.stylesheet.layers); function getbuckets(buckets, ordered, layers) { for (var a = 0; a < layers.length; a++) { var layer = layers[a]; if (layer.layers) { buckets = getbuckets(buckets, ordered, layer.layers); } if (!layer.source || !layer.type) { continue; } var bucket = { id: layer.id }; for (var prop in layer) { if ((/^style/).test(prop)) continue; bucket[prop] = layer[prop]; } bucket.render = StyleConstant.resolve(bucket.render, constants); buckets[layer.id] = bucket; ordered.push(bucket); } return buckets; } // style class keys var styleNames = ['style']; for (className in this.classes) styleNames.push('style.' + className); // apply layer group inheritance resulting in a flattened array var flattened = flattenLayers(this.stylesheet.layers); // map layer ids to layer definitions for resolving refs var layermap = this.layermap = {}; for (a = 0; a < flattened.length; a++) { layer = flattened[a]; var newLayer = {}; for (var k in layer) { if (k === 'layers') continue; newLayer[k] = layer[k]; } layermap[layer.id] = newLayer; flattened[a] = newLayer; } for (a = 0; a < flattened.length; a++) { flattened[a] = resolveLayer(layermap, flattened[a]); } // Resolve layer references. function resolveLayer(layermap, layer) { if (!layer.ref || !layermap[layer.ref]) return layer; var parent = resolveLayer(layermap, layermap[layer.ref]); layer.render = parent.render; layer.type = parent.type; layer.filter = parent.filter; layer.source = parent.source; layer['source-layer'] = parent['source-layer']; layer['min-zoom'] = parent['min-zoom']; layer['max-zoom'] = parent['max-zoom']; return layer; } // Flatten composite layer structures. function flattenLayers(layers) { var flat = []; for (var i = 0; i < layers.length; i++) { flat.push(layers[i]); if (layers[i].layers) { flat.push.apply(flat, flattenLayers(layers[i].layers)); } } return flat; } var transitions = {}; var globalTrans = this.stylesheet.transition; for (a in flattened) { layer = flattened[a]; id = layer.id; style = {}; styleTrans = {}; // basic cascading of styles for (b = 0; b < styleNames.length; b++) { styleName = styleNames[b]; if (!layer[styleName]) continue; // set style properties for (prop in layer[styleName]) { if (prop.indexOf('transition-') === -1) { style[prop] = layer[styleName][prop]; } else { styleTrans[prop.replace('transition-', '')] = layer[styleName][prop]; } } } style = StyleConstant.resolve(style, constants); var renderType = layer.type; transitions[id] = {}; for (prop in style) { var newDeclaration = new StyleDeclaration(renderType, prop, style[prop]); var oldTransition = this.transitions[id] && this.transitions[id][prop]; var newStyleTrans = {}; newStyleTrans.duration = styleTrans[prop] && styleTrans[prop].duration ? styleTrans[prop].duration : globalTrans && globalTrans.duration ? globalTrans.duration : 300; newStyleTrans.delay = styleTrans[prop] && styleTrans[prop].delay ? styleTrans[prop].delay : globalTrans && globalTrans.delay ? globalTrans.delay : 0; if (!options.transition) { newStyleTrans.duration = 0; newStyleTrans.delay = 0; } // Only create a new transition if the declaration changed if (!oldTransition || oldTransition.declaration.json !== newDeclaration.json) { var newTransition = new StyleTransition(newDeclaration, oldTransition, newStyleTrans); transitions[id][prop] = newTransition; // Run the animation loop until the end of the transition if (!newTransition.instant()) { newTransition.loopID = this.animationLoop.set(newTransition.endTime - (new Date()).getTime()); } if (oldTransition) { this.animationLoop.cancel(oldTransition.loopID); } } else { transitions[id][prop] = oldTransition; } } } this.transitions = transitions; this.layerGroups = this._groupLayers(this.stylesheet.layers); this.fire('change'); }; /* This should be moved elsewhere. Localizing resources doesn't belong here */ Style.prototype.setSprite = function(sprite) { this.sprite = new ImageSprite(sprite); this.sprite.on('loaded', this.fire.bind(this, 'change')); }; // Modify classes Style.prototype.addClass = function(n, options) { if (this.classes[n]) return; // prevent unnecessary recalculation this.classes[n] = true; this.cascade(options); }; Style.prototype.removeClass = function(n, options) { if (!this.classes[n]) return; // prevent unnecessary recalculation delete this.classes[n]; this.cascade(options); }; Style.prototype.hasClass = function(n) { return !!this.classes[n]; }; Style.prototype.setClassList = function(l, options) { this.classes = {}; for (var i = 0; i < l.length; i++) { this.classes[l[i]] = true; } this.cascade(options); }; Style.prototype.getClassList = function() { return Object.keys(this.classes); }; Style.prototype.getLayer = function(id) { return this.layermap[id]; }; },{"../util/evented.js":82,"../util/util.js":87,"./calculatedstyle.js":50,"./imagesprite.js":51,"./styleconstant.js":54,"./styledeclaration.js":55,"./styletransition.js":56}],54:[function(require,module,exports){ 'use strict'; var util = require('../util/util.js'); module.exports.resolve = function (properties, constants) { if (!constants) return properties; var result = {}, i; function resolve(value) { return typeof value === 'string' && value[0] === '@' ? constants[value] : value; } for (var key in properties) { var value = resolve(properties[key]); if (Array.isArray(value)) { value = value.slice(); for (i = 0; i < value.length; i++) { if (value[i] in constants) { value[i] = resolve(value[i]); } } } if (value.stops) { value = util.extend({}, value); value.stops = value.stops.slice(); for (i = 0; i < value.stops.length; i++) { if (value.stops[i][1] in constants) { value.stops[i] = [ value.stops[i][0], resolve(value.stops[i][1]) ]; } } } result[key] = value; } return result; }; },{"../util/util.js":87}],55:[function(require,module,exports){ 'use strict'; var util = require('../util/util.js'), reference = require('mapbox-gl-style-spec/reference/v4'), parseCSSColor = require('csscolorparser').parseCSSColor; module.exports = StyleDeclaration; /* * A parsed representation of a property:value pair */ function StyleDeclaration(renderType, prop, value) { var className = 'class_' + renderType; var propReference = reference[className] && reference[className][prop]; if (!propReference) return; this.value = this.parseValue(value, propReference.type, propReference.values); this.prop = prop; this.type = propReference.type; // immuatable representation of value. used for comparison this.json = JSON.stringify(value); } StyleDeclaration.prototype.calculate = function(z) { return typeof this.value === 'function' ? this.value(z) : this.value; }; StyleDeclaration.prototype.parseValue = function(value, type, values) { if (type === 'color') { return parseColor(value); } else if (type === 'number') { return parseNumber(value); } else if (type === 'boolean') { return Boolean(value); } else if (type === 'image') { return String(value); } else if (type === 'string') { return String(value); } else if (type === 'array') { return parseNumberArray(value); } else if (type === 'enum' && Array.isArray(values)) { return values.indexOf(value) >= 0 ? value : undefined; } else { console.warn(type + ' is not a supported property type'); } }; function parseNumber(num) { if (num.stops) num = stopsFn(num); var value = +num; return !isNaN(value) ? value : num; } function parseNumberArray(array) { var widths = array.map(parseNumber); return function(z) { var result = []; for (var i = 0; i < widths.length; i++) { result.push(typeof widths[i] === 'function' ? widths[i](z) : widths[i]); } return result; }; } var colorCache = {}; function parseColor(value) { if (value.stops) { for (var i = 0; i < value.stops.length; i++) { // store the parsed color as the 3rd element in the array value.stops[i][2] = parseCSSColor(value.stops[i][1]); } return stopsFn(value, true); } if (colorCache[value]) { return colorCache[value]; } var color = prepareColor(parseCSSColor(value)); colorCache[value] = color; return color; } function stopsFn(params, color) { var stops = params.stops; var base = params.base || reference.function.base.default; return function(z) { // find the two stops which the current z is between var low = null; var high = null; for (var i = 0; i < stops.length; i++) { var stop = stops[i]; if (stop[0] <= z) low = stop; if (stop[0] > z) { high = stop; break; } } if (low && high) { var zoomDiff = high[0] - low[0]; var zoomProgress = z - low[0]; var t = 0; if (base == 1) { t = zoomProgress / zoomDiff; } else { t = (Math.pow(base, zoomProgress) - 1) / (Math.pow(base, zoomDiff) - 1); } if (color) return prepareColor(interpColor(low[2], high[2], t)); else return util.interp(low[1], high[1], t); } else if (low) { if (color) return prepareColor(low[2]); else return low[1]; } else if (high) { if (color) return prepareColor(high[2]); else return high[1]; } else { if (color) return [0, 0, 0, 1]; else return 1; } }; } function prepareColor(c) { return [c[0] / 255, c[1] / 255, c[2] / 255, c[3] / 1]; } function interpColor(from, to, t) { return [ util.interp(from[0], to[0], t), util.interp(from[1], to[1], t), util.interp(from[2], to[2], t), util.interp(from[3], to[3], t) ]; } },{"../util/util.js":87,"csscolorparser":91,"mapbox-gl-style-spec/reference/v4":95}],56:[function(require,module,exports){ 'use strict'; var util = require('../util/util.js'); module.exports = StyleTransition; /* * Represents a transition between two declarations */ function StyleTransition(declaration, oldTransition, value) { this.declaration = declaration; this.startTime = this.endTime = (new Date()).getTime(); var type = declaration.type; if (type === 'number') { this.interp = util.interp; } else if (type === 'color') { this.interp = interpColor; } else if (type === 'array') { this.interp = interpNumberArray; } this.oldTransition = oldTransition; this.duration = value.duration || 0; this.delay = value.delay || 0; if (!this.instant()) { this.endTime = this.startTime + this.duration + this.delay; this.ease = util.easeCubicInOut; } if (oldTransition && oldTransition.endTime <= this.startTime) { // Old transition is done running, so we can // delete its reference to its old transition. delete oldTransition.oldTransition; } } StyleTransition.prototype.instant = function() { return !this.oldTransition || !this.interp || (this.duration === 0 && this.delay === 0); }; /* * Return the value of the transitioning property at zoom level `z` and optional time `t` */ StyleTransition.prototype.at = function(z, t) { var value = this.declaration.calculate(z); if (this.instant()) return value; t = t || Date.now(); if (t < this.endTime) { var oldValue = this.oldTransition.at(z, this.startTime); var eased = this.ease((t - this.startTime - this.delay) / this.duration); value = this.interp(oldValue, value, eased); } return value; }; function interpNumberArray(from, to, t) { return from.map(function(d, i) { return util.interp(d, to[i], t); }); } function interpColor(from, to, t) { return [ util.interp(from[0], to[0], t), util.interp(from[1], to[1], t), util.interp(from[2], to[2], t), util.interp(from[3], to[3], t) ]; } },{"../util/util.js":87}],57:[function(require,module,exports){ 'use strict'; var Point = require('point-geometry'); module.exports = Anchor; function Anchor(x, y, angle, scale, segment) { this.x = x; this.y = y; this.angle = angle; this.scale = scale; if (segment !== undefined) { this.segment = segment; } } Anchor.prototype = Object.create(Point.prototype); Anchor.prototype.clone = function() { return new Anchor(this.x, this.y, this.angle, this.scale, this.segment); }; },{"point-geometry":98}],58:[function(require,module,exports){ 'use strict'; module.exports = BinPack; function BinPack(width, height) { this.width = width; this.height = height; this.free = [{ x: 0, y: 0, w: width, h: height }]; } BinPack.prototype.release = function(rect) { // Simple algorithm to recursively merge the newly released cell with its // neighbor. This doesn't merge more than two cells at a time, and fails // for complicated merges. for (var i = 0; i < this.free.length; i++) { var free = this.free[i]; if (free.y == rect.y && free.h == rect.h && free.x + free.w == rect.x) { free.w += rect.w; } else if (free.x == rect.x && free.w == rect.w && free.y + free.h == rect.y) { free.h += rect.h; } else if (rect.y == free.y && rect.h == free.h && rect.x + rect.w == free.x) { free.x = rect.x; free.w += rect.w; } else if (rect.x == free.x && rect.w == free.w && rect.y + rect.h == free.y) { free.y = rect.y; free.h += rect.h; } else { continue; } this.free.splice(i, 1); this.release(free); return; } this.free.push(rect); }; BinPack.prototype.allocate = function(width, height) { // Find the smallest free rect angle var rect = { x: Infinity, y: Infinity, w: Infinity, h: Infinity }; var smallest = -1; for (var i = 0; i < this.free.length; i++) { var ref = this.free[i]; if (width <= ref.w && height <= ref.h && ref.y <= rect.y && ref.x <= rect.x) { rect = ref; smallest = i; } } if (smallest < 0) { // There's no space left for this char. return { x: -1, y: -1 }; } else { this.free.splice(smallest, 1); // Shorter/Longer Axis Split Rule (SAS) // http://clb.demon.fi/files/RectangleBinPack.pdf p. 15 // Ignore the dimension of R and just split long the shorter dimension // See Also: http://www.cs.princeton.edu/~chazelle/pubs/blbinpacking.pdf if (rect.w < rect.h) { // split horizontally // +--+---+ // |__|___| <-- b1 // +------+ <-- b2 if (rect.w > width) this.free.push({ x: rect.x + width, y: rect.y, w: rect.w - width, h: height }); if (rect.h > height) this.free.push({ x: rect.x, y: rect.y + height, w: rect.w, h: rect.h - height }); } else { // split vertically // +--+---+ // |__| | <-- b1 // +--|---+ <-- b2 if (rect.w > width) this.free.push({ x: rect.x + width, y: rect.y, w: rect.w - width, h: rect.h }); if (rect.h > height) this.free.push({ x: rect.x, y: rect.y + height, w: width, h: rect.h - height }); } return { x: rect.x, y: rect.y, w: width, h: height }; } }; },{}],59:[function(require,module,exports){ 'use strict'; var rbush = require('rbush'), rotationRange = require('./rotationrange.js'), Point = require('point-geometry'); module.exports = Collision; function Collision(zoom, tileExtent, tileSize, placementDepth) { this.hTree = rbush(); // tree for horizontal labels this.cTree = rbush(); // tree for glyphs from curved labels // tile pixels per screen pixels at the tile's zoom level this.tilePixelRatio = tileExtent / tileSize; this.zoom = zoom; // Calculate the maximum scale we can go down in our fake-3d rtree so that // placement still makes sense. This is calculated so that the minimum // placement zoom can be at most 25.5 (we use an unsigned integer x10 to // store the minimum zoom). // // We don't want to place labels all the way to 25.5. This lets too many // glyphs be placed, slowing down collision checking. Only place labels if // they will show up within the intended zoom range of the tile. placementDepth = Math.min(3, placementDepth || 1, 25.5 - this.zoom); this.maxPlacementScale = Math.exp(Math.LN2 * placementDepth); var m = 4096; var edge = m * this.tilePixelRatio * 2; var fullRange = [Math.PI * 2, 0]; this.left = { anchor: new Point(0, 0), box: { x1: -edge, y1: -edge, x2: 0, y2: edge }, placementRange: fullRange, placementScale: 0.5, maxScale: Infinity, padding: 0 }; this.top = { anchor: new Point(0, 0), box: { x1: -edge, y1: -edge, x2: edge, y2: 0 }, placementRange: fullRange, placementScale: 0.5, maxScale: Infinity, padding: 0 }; this.bottom = { anchor: new Point(m, m), box: { x1: -edge, y1: 0, x2: edge, y2: edge }, placementRange: fullRange, placementScale: 0.5, maxScale: Infinity, padding: 0 }; this.right = { anchor: new Point(m, m), box: { x1: 0, y1: -edge, x2: edge, y2: edge }, placementRange: fullRange, placementScale: 0.5, maxScale: Infinity, padding: 0 }; } Collision.prototype.getPlacementScale = function(glyphs, minPlacementScale, avoidEdges) { var left = this.left; var right = this.right; var top = this.top; var bottom = this.bottom; for (var k = 0; k < glyphs.length; k++) { var glyph = glyphs[k]; var box = glyph.box; var bbox = glyph.hBox || box; var anchor = glyph.anchor; var pad = glyph.padding; var minScale = Math.max(minPlacementScale, glyph.minScale); var maxScale = glyph.maxScale || Infinity; if (minScale >= maxScale) continue; // Compute the scaled bounding box of the unrotated glyph var searchBox = this.getBox(anchor, bbox, minScale, maxScale); var blocking = this.hTree.search(searchBox).concat(this.cTree.search(searchBox)); if (avoidEdges) { if (searchBox[0] < 0) blocking.push(left); if (searchBox[1] < 0) blocking.push(top); if (searchBox[2] >= 4096) blocking.push(right); if (searchBox[3] >= 4096) blocking.push(bottom); } if (blocking.length) { var na = anchor; // new anchor var nb = box; // new box for (var l = 0; l < blocking.length; l++) { var oa = blocking[l].anchor; // old anchor var ob = blocking[l].box; // old box // If anchors are identical, we're going to skip the label. // NOTE: this isn't right because there can be glyphs with // the same anchor but differing box offsets. if (na.equals(oa)) { return null; } // todo: unhardcode the 8 = tileExtent/tileSize var padding = Math.max(pad, blocking[l].padding) * 8; // Original algorithm: var s1 = (ob.x1 - nb.x2 - padding) / (na.x - oa.x); // scale at which new box is to the left of old box var s2 = (ob.x2 - nb.x1 + padding) / (na.x - oa.x); // scale at which new box is to the right of old box var s3 = (ob.y1 - nb.y2 - padding) / (na.y - oa.y); // scale at which new box is to the top of old box var s4 = (ob.y2 - nb.y1 + padding) / (na.y - oa.y); // scale at which new box is to the bottom of old box if (isNaN(s1) || isNaN(s2)) s1 = s2 = 1; if (isNaN(s3) || isNaN(s4)) s3 = s4 = 1; var collisionFreeScale = Math.min(Math.max(s1, s2), Math.max(s3, s4)); // Only update label's min scale if the glyph was restricted by a collision if (collisionFreeScale > minPlacementScale && collisionFreeScale > minScale && collisionFreeScale < maxScale && collisionFreeScale < blocking[l].maxScale) { minPlacementScale = collisionFreeScale; } if (minPlacementScale > this.maxPlacementScale) { return null; } } } } return minPlacementScale; }; Collision.prototype.getPlacementRange = function(glyphs, placementScale, horizontal) { var placementRange = [2*Math.PI, 0]; for (var k = 0; k < glyphs.length; k++) { var glyph = glyphs[k]; var bbox = glyph.hBox || glyph.box; var anchor = glyph.anchor; var minPlacedX = anchor.x + bbox.x1 / placementScale; var minPlacedY = anchor.y + bbox.y1 / placementScale; var maxPlacedX = anchor.x + bbox.x2 / placementScale; var maxPlacedY = anchor.y + bbox.y2 / placementScale; var searchBox = [minPlacedX, minPlacedY, maxPlacedX, maxPlacedY]; var blocking = this.hTree.search(searchBox); if (horizontal) { blocking = blocking.concat(this.cTree.search(searchBox)); } for (var l = 0; l < blocking.length; l++) { var b = blocking[l]; var bbox2 = b.hBox || b.box; var x1, x2, y1, y2, intersectX, intersectY; // Adjust and compare bboxes to see if the glyphs might intersect if (placementScale > b.placementScale) { x1 = b.anchor.x + bbox2.x1 / placementScale; y1 = b.anchor.y + bbox2.y1 / placementScale; x2 = b.anchor.x + bbox2.x2 / placementScale; y2 = b.anchor.y + bbox2.y2 / placementScale; intersectX = x1 < maxPlacedX && x2 > minPlacedX; intersectY = y1 < maxPlacedY && y2 > minPlacedY; } else { x1 = anchor.x + bbox.x1 / b.placementScale; y1 = anchor.y + bbox.y1 / b.placementScale; x2 = anchor.x + bbox.x2 / b.placementScale; y2 = anchor.y + bbox.y2 / b.placementScale; intersectX = x1 < b[2] && x2 > b[0]; intersectY = y1 < b[3] && y2 > b[1]; } // If they can't intersect, skip more expensive rotation calculation if (!(intersectX && intersectY)) continue; var scale = Math.max(placementScale, b.placementScale); var range = rotationRange.rotationRange(glyph, b, scale); placementRange[0] = Math.min(placementRange[0], range[0]); placementRange[1] = Math.max(placementRange[1], range[1]); } } return placementRange; }; // Insert glyph placements into rtree. Collision.prototype.insert = function(glyphs, anchor, placementScale, placementRange, horizontal) { var allBounds = []; for (var k = 0; k < glyphs.length; k++) { var glyph = glyphs[k]; var bbox = glyph.hBox || glyph.box; var minScale = Math.max(placementScale, glyph.minScale); var maxScale = glyph.maxScale || Infinity; var bounds = this.getBox(anchor, bbox, minScale, maxScale); bounds.anchor = anchor; bounds.box = glyph.box; if (glyph.hBox) bounds.hBox = bbox; bounds.placementRange = placementRange; bounds.placementScale = minScale; bounds.maxScale = maxScale; bounds.padding = glyph.padding; allBounds.push(bounds); } (horizontal ? this.hTree : this.cTree).load(allBounds); }; Collision.prototype.getBox = function(anchor, bbox, minScale, maxScale) { return [ anchor.x + Math.min(bbox.x1 / minScale, bbox.x1 / maxScale), anchor.y + Math.min(bbox.y1 / minScale, bbox.y1 / maxScale), anchor.x + Math.max(bbox.x2 / minScale, bbox.x2 / maxScale), anchor.y + Math.max(bbox.y2 / minScale, bbox.y2 / maxScale)]; }; },{"./rotationrange.js":65,"point-geometry":98,"rbush":100}],60:[function(require,module,exports){ 'use strict'; var BinPack = require('./binpack.js'); module.exports = GlyphAtlas; function GlyphAtlas(width, height) { this.width = width; this.height = height; this.bin = new BinPack(width, height); this.index = {}; this.ids = {}; this.data = new Uint8Array(width * height); } GlyphAtlas.prototype = { get debug() { return 'canvas' in this; }, set debug(value) { if (value && !this.canvas) { this.canvas = document.createElement('canvas'); this.canvas.width = this.width; this.canvas.height = this.height; document.body.appendChild(this.canvas); this.ctx = this.canvas.getContext('2d'); } else if (!value && this.canvas) { this.canvas.parentNode.removeChild(this.canvas); delete this.ctx; delete this.canvas; } } }; GlyphAtlas.prototype.getGlyphs = function() { var glyphs = {}, split, name, id; for (var key in this.ids) { split = key.split('#'); name = split[0]; id = split[1]; if (!glyphs[name]) glyphs[name] = []; glyphs[name].push(id); } return glyphs; }; GlyphAtlas.prototype.getRects = function() { var rects = {}, split, name, id; for (var key in this.ids) { split = key.split('#'); name = split[0]; id = split[1]; if (!rects[name]) rects[name] = {}; rects[name][id] = this.index[key]; } return rects; }; GlyphAtlas.prototype.removeGlyphs = function(id) { for (var key in this.ids) { var ids = this.ids[key]; var pos = ids.indexOf(id); if (pos >= 0) ids.splice(pos, 1); this.ids[key] = ids; if (!ids.length) { var rect = this.index[key]; var target = this.data; for (var y = 0; y < rect.h; y++) { var y1 = this.width * (rect.y + y) + rect.x; for (var x = 0; x < rect.w; x++) { target[y1 + x] = 0; } } this.dirty = true; this.bin.release(rect); delete this.index[key]; delete this.ids[key]; } } this.updateTexture(this.gl); }; GlyphAtlas.prototype.addGlyph = function(id, name, glyph, buffer) { if (!glyph) { // console.warn('missing glyph', code, String.fromCharCode(code)); return null; } var key = name + "#" + glyph.id; // The glyph is already in this texture. if (this.index[key]) { if (this.ids[key].indexOf(id) < 0) { this.ids[key].push(id); } return this.index[key]; } // The glyph bitmap has zero width. if (!glyph.bitmap) { return null; } var buffered_width = glyph.width + buffer * 2; var buffered_height = glyph.height + buffer * 2; // Add a 1px border around every image. var pack_width = buffered_width; var pack_height = buffered_height; // Increase to next number divisible by 4, but at least 1. // This is so we can scale down the texture coordinates and pack them // into 2 bytes rather than 4 bytes. pack_width += (4 - pack_width % 4); pack_height += (4 - pack_height % 4); var rect = this.bin.allocate(pack_width, pack_height); if (rect.x < 0) { console.warn('glyph bitmap overflow'); return { glyph: glyph, rect: null }; } // Add left and top glyph offsets to rect. rect.l = glyph.left; rect.t = glyph.top; this.index[key] = rect; this.ids[key] = [id]; var target = this.data; var source = glyph.bitmap; for (var y = 0; y < buffered_height; y++) { var y1 = this.width * (rect.y + y) + rect.x; var y2 = buffered_width * y; for (var x = 0; x < buffered_width; x++) { target[y1 + x] = source[y2 + x]; } } this.dirty = true; return rect; }; GlyphAtlas.prototype.bind = function(gl) { this.gl = gl; if (!this.texture) { this.texture = gl.createTexture(); gl.bindTexture(gl.TEXTURE_2D, this.texture); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); gl.texImage2D(gl.TEXTURE_2D, 0, gl.ALPHA, this.width, this.height, 0, gl.ALPHA, gl.UNSIGNED_BYTE, null); } else { gl.bindTexture(gl.TEXTURE_2D, this.texture); } }; GlyphAtlas.prototype.updateTexture = function(gl) { this.bind(gl); if (this.dirty) { gl.texSubImage2D(gl.TEXTURE_2D, 0, 0, 0, this.width, this.height, gl.ALPHA, gl.UNSIGNED_BYTE, this.data); // DEBUG if (this.ctx) { var data = this.ctx.getImageData(0, 0, this.width, this.height); for (var i = 0, j = 0; i < this.data.length; i++, j += 4) { data.data[j] = this.data[i]; data.data[j+1] = this.data[i]; data.data[j+2] = this.data[i]; data.data[j+3] = 255; } this.ctx.putImageData(data, 0, 0); this.ctx.strokeStyle = 'red'; for (var k = 0; k < this.bin.free.length; k++) { var free = this.bin.free[k]; this.ctx.strokeRect(free.x, free.y, free.w, free.h); } } // END DEBUG this.dirty = false; } }; },{"./binpack.js":58}],61:[function(require,module,exports){ 'use strict'; var getArrayBuffer = require('../util/ajax.js').getArrayBuffer; var Glyphs = require('../util/glyphs.js'); var Protobuf = require('pbf'); module.exports = GlyphSource; function GlyphSource(url, glyphAtlas) { this.url = url; this.glyphAtlas = glyphAtlas; this.stacks = {}; this.loading = {}; } GlyphSource.prototype.getRects = function(fontstack, glyphIDs, tileID, callback) { if (this.stacks[fontstack] === undefined) this.stacks[fontstack] = {}; var rects = {}; var glyphs = {}; var result = { rects: rects, glyphs: glyphs }; var stack = this.stacks[fontstack]; var glyphAtlas = this.glyphAtlas; var missing = {}; var remaining = 0; for (var i = 0; i < glyphIDs.length; i++) { var glyphID = glyphIDs[i]; var range = Math.floor(glyphID / 256); if (stack[range]) { var glyph = stack[range].glyphs[glyphID]; var buffer = 3; rects[glyphID] = glyphAtlas.addGlyph(tileID, fontstack, glyph, buffer); if (glyph) glyphs[glyphID] = simpleGlyph(glyph); } else { if (missing[range] === undefined) { missing[range] = []; remaining++; } missing[range].push(glyphID); } } if (!remaining) callback(undefined, result); var glyphSource = this; for (var r in missing) { this.loadRange(fontstack, r, onRangeLoaded); } function onRangeLoaded(err, range, data) { // TODO not be silent about errors if (!err) { var stack = glyphSource.stacks[fontstack][range] = data.stacks[fontstack]; for (var i = 0; i < missing[range].length; i++) { var glyphID = missing[range][i]; var glyph = stack.glyphs[glyphID]; var buffer = 3; rects[glyphID] = glyphAtlas.addGlyph(tileID, fontstack, glyph, buffer); if (glyph) glyphs[glyphID] = simpleGlyph(glyph); } } remaining--; if (!remaining) callback(undefined, result); } }; function simpleGlyph(glyph) { return { advance: glyph.advance, left: glyph.left, top: glyph.top }; } GlyphSource.prototype.loadRange = function(fontstack, range, callback) { if (range * 256 >= 65280) return callback('gyphs > 65280 not supported'); if (this.loading[fontstack] === undefined) this.loading[fontstack] = {}; var loading = this.loading[fontstack]; if (loading[range]) { loading[range].push(callback); } else { loading[range] = [callback]; var rangeName = (range * 256) + '-' + (range * 256 + 255); var url = glyphUrl(fontstack, rangeName, this.url); getArrayBuffer(url, function(err, data) { var glyphs = !err && new Glyphs(new Protobuf(new Uint8Array(data))); for (var i = 0; i < loading[range].length; i++) { loading[range][i](err, range, glyphs); } delete loading[range]; }); } }; function glyphUrl(fontstack, range, url, subdomains) { subdomains = subdomains || 'abc'; return url .replace('{s}', subdomains[fontstack.length % subdomains.length]) .replace('{fontstack}', fontstack) .replace('{range}', range); } },{"../util/ajax.js":76,"../util/glyphs.js":83,"pbf":96}],62:[function(require,module,exports){ 'use strict'; var util = require('../util/util.js'), Anchor = require('../symbol/anchor.js'); module.exports = interpolate; var minScale = 0.5; var minScaleArrays = { 1: [minScale], 2: [minScale, 2], 4: [minScale, 4, 2, 4], 8: [minScale, 8, 4, 8, 2, 8, 4, 8] }; function interpolate(vertices, spacing, minScale, maxScale, tilePixelRatio, start) { if (minScale === undefined) minScale = 0; maxScale = Math.round(Math.max(Math.min(8, maxScale / 2), 1)); spacing *= tilePixelRatio / maxScale; var minScales = minScaleArrays[maxScale]; var len = minScales.length; var distance = 0, markedDistance = 0, added = start || 0; var points = []; for (var i = 0; i < vertices.length - 1; i++) { var a = vertices[i], b = vertices[i + 1]; var segmentDist = a.dist(b), angle = b.angleTo(a); while (markedDistance + spacing < distance + segmentDist) { markedDistance += spacing; var t = (markedDistance - distance) / segmentDist, x = util.interp(a.x, b.x, t), y = util.interp(a.y, b.y, t), s = minScales[added % len]; if (x >= 0 && x < 4096 && y >= 0 && y < 4096) { points.push(new Anchor(x, y, angle, s, i)); } added++; } distance += segmentDist; } return points; } },{"../symbol/anchor.js":57,"../util/util.js":87}],63:[function(require,module,exports){ 'use strict'; var Point = require('point-geometry'); module.exports = { getIcon: getIcon, getGlyphs: getGlyphs }; var minScale = 0.5; // underscale by 1 zoom level function getIcon(anchor, image, boxScale, line, props) { var x = image.width / 2 / image.pixelRatio; var y = image.height / 2 / image.pixelRatio; var dx = props['icon-offset'][0]; var dy = props['icon-offset'][1]; var x1 = (dx - x); var x2 = (dx + x); var y1 = (dy - y); var y2 = (dy + y); var tl = new Point(x1, y1); var tr = new Point(x2, y1); var br = new Point(x2, y2); var bl = new Point(x1, y2); var angle = props['icon-rotate'] * Math.PI / 180; if (anchor.segment !== undefined && props['icon-rotation-alignment'] !== 'viewport') { var next = line[anchor.segment]; angle += -Math.atan2(next.x - anchor.x, next.y - anchor.y) + Math.PI / 2; } if (angle) { var sin = Math.sin(angle), cos = Math.cos(angle), matrix = [cos, -sin, sin, cos]; tl = tl.matMult(matrix); tr = tr.matMult(matrix); bl = bl.matMult(matrix); br = br.matMult(matrix); x1 = Math.min(tl.x, tr.x, bl.x, br.x); x2 = Math.max(tl.x, tr.x, bl.x, br.x); y1 = Math.min(tl.y, tr.y, bl.y, br.y); y2 = Math.max(tl.y, tr.y, bl.y, br.y); } var box = { x1: x1 * boxScale, x2: x2 * boxScale, y1: y1 * boxScale, y2: y2 * boxScale }; var iconBox = { box: box, anchor: anchor, minScale: minScale, maxScale: Infinity, padding: props['icon-padding'] }; var icon = { tl: tl, tr: tr, br: br, bl: bl, tex: image, angle: 0, anchor: anchor, minScale: minScale, maxScale: Infinity }; return { shapes: [icon], boxes: [iconBox], minScale: anchor.scale }; } function getGlyphs(anchor, origin, shaping, faces, boxScale, horizontal, line, props) { var maxAngleDelta = props['text-max-angle'] * Math.PI / 180; var rotate = props['text-rotate'] * Math.PI / 180; var padding = props['text-padding']; var alongLine = props['text-rotation-alignment'] !== 'viewport'; var keepUpright = props['text-keep-upright']; var glyphs = [], boxes = []; var buffer = 3; for (var k = 0; k < shaping.length; k++) { var shape = shaping[k]; var fontstack = faces[shape.fontstack]; var glyph = fontstack.glyphs[shape.glyph]; var rect = fontstack.rects[shape.glyph]; if (!glyph) continue; if (!(rect && rect.w > 0 && rect.h > 0)) continue; var x = (origin.x + shape.x + glyph.left - buffer + rect.w / 2) * boxScale; var glyphInstances; if (anchor.segment !== undefined && alongLine) { glyphInstances = []; getSegmentGlyphs(glyphInstances, anchor, x, line, anchor.segment, 1, maxAngleDelta); if (keepUpright) getSegmentGlyphs(glyphInstances, anchor, x, line, anchor.segment, -1, maxAngleDelta); } else { glyphInstances = [{ anchor: anchor, offset: 0, angle: 0, maxScale: Infinity, minScale: minScale }]; } var x1 = origin.x + shape.x + glyph.left - buffer, y1 = origin.y + shape.y - glyph.top - buffer, x2 = x1 + rect.w, y2 = y1 + rect.h, otl = new Point(x1, y1), otr = new Point(x2, y1), obl = new Point(x1, y2), obr = new Point(x2, y2); var obox = { x1: boxScale * x1, y1: boxScale * y1, x2: boxScale * x2, y2: boxScale * y2 }; for (var i = 0; i < glyphInstances.length; i++) { var instance = glyphInstances[i], tl = otl, tr = otr, bl = obl, br = obr, box = obox, // Clamp to -90/+90 degrees angle = instance.angle + rotate; if (angle) { // Compute the transformation matrix. var sin = Math.sin(angle), cos = Math.cos(angle), matrix = [cos, -sin, sin, cos]; tl = tl.matMult(matrix); tr = tr.matMult(matrix); bl = bl.matMult(matrix); br = br.matMult(matrix); } // Prevent label from extending past the end of the line var glyphMinScale = Math.max(instance.minScale, anchor.scale); // Remember the glyph for later insertion. glyphs.push({ tl: tl, tr: tr, bl: bl, br: br, tex: rect, angle: (anchor.angle + rotate + instance.offset + 2 * Math.PI) % (2 * Math.PI), anchor: instance.anchor, minScale: glyphMinScale, maxScale: instance.maxScale }); if (!instance.offset) { // not a flipped glyph if (angle) { // Calculate the rotated glyph's bounding box offsets from the anchor point. box = { x1: boxScale * Math.min(tl.x, tr.x, bl.x, br.x), y1: boxScale * Math.min(tl.y, tr.y, bl.y, br.y), x2: boxScale * Math.max(tl.x, tr.x, bl.x, br.x), y2: boxScale * Math.max(tl.y, tr.y, bl.y, br.y) }; } boxes.push({ box: box, anchor: instance.anchor, minScale: glyphMinScale, maxScale: instance.maxScale, padding: padding }); } } } // TODO avoid creating the boxes in the first place? if (horizontal) boxes = [getMergedBoxes(boxes, anchor)]; var minPlacementScale = anchor.scale; var minGlyphScale = Infinity; for (var m = 0; m < boxes.length; m++) { minGlyphScale = Math.min(minGlyphScale, boxes[m].minScale); } minGlyphScale = Math.max(minPlacementScale, minScale); return { boxes: boxes, shapes: glyphs, minScale: minGlyphScale }; } function getSegmentGlyphs(glyphs, anchor, offset, line, segment, direction, maxAngleDelta) { var upsideDown = direction < 0; if (offset < 0) direction *= -1; if (direction > 0) segment++; var newAnchor = anchor; var end = line[segment]; var prevscale = Infinity; var prevAngle; offset = Math.abs(offset); var placementScale = anchor.scale; segment_loop: while (true) { var dist = newAnchor.dist(end); var scale = offset/dist; var angle = -Math.atan2(end.x - newAnchor.x, end.y - newAnchor.y) + direction * Math.PI / 2; if (upsideDown) angle += Math.PI; // Don't place around sharp corners var angleDiff = (angle - prevAngle) % (2 * Math.PI); if (prevAngle && Math.abs(angleDiff) > maxAngleDelta) { anchor.scale = prevscale; break; } glyphs.push({ anchor: newAnchor, offset: upsideDown ? Math.PI : 0, minScale: scale, maxScale: prevscale, angle: (angle + 2 * Math.PI) % (2 * Math.PI) }); if (scale <= placementScale) break; newAnchor = end; // skip duplicate nodes while (newAnchor.equals(end)) { segment += direction; end = line[segment]; if (!end) { anchor.scale = scale; break segment_loop; } } var unit = end.sub(newAnchor)._unit(); newAnchor = newAnchor.sub(unit._mult(dist)); prevscale = scale; prevAngle = angle; } } function getMergedBoxes(glyphs, anchor) { // Collision checks between rotating and fixed labels are relatively expensive, // so we use one box per label, not per glyph for horizontal labels. var mergedglyphs = { box: { x1: Infinity, y1: Infinity, x2: -Infinity, y2: -Infinity }, anchor: anchor, minScale: 0, padding: -Infinity }; var box = mergedglyphs.box; for (var m = 0; m < glyphs.length; m++) { var gbox = glyphs[m].box; box.x1 = Math.min(box.x1, gbox.x1); box.y1 = Math.min(box.y1, gbox.y1); box.x2 = Math.max(box.x2, gbox.x2); box.y2 = Math.max(box.y2, gbox.y2); mergedglyphs.minScale = Math.max(mergedglyphs.minScale, glyphs[m].minScale); mergedglyphs.padding = Math.max(mergedglyphs.padding, glyphs[m].padding); } // for all horizontal labels, calculate bbox covering all rotated positions var x12 = box.x1 * box.x1, y12 = box.y1 * box.y1, x22 = box.x2 * box.x2, y22 = box.y2 * box.y2, diag = Math.sqrt(Math.max(x12 + y12, x12 + y22, x22 + y12, x22 + y22)); mergedglyphs.hBox = { x1: -diag, y1: -diag, x2: diag, y2: diag }; return mergedglyphs; } },{"point-geometry":98}],64:[function(require,module,exports){ 'use strict'; var resolveTokens = require('../util/token.js'); module.exports = resolveText; // For an array of features determine what glyph ranges need to be loaded // and apply any text preprocessing. The remaining users of text should // use the `textFeatures` key returned by this function rather than accessing // feature text directly. function resolveText(features, info, glyphs) { var textFeatures = []; var codepoints = []; for (var i = 0, fl = features.length; i < fl; i++) { var text = resolveTokens(features[i].properties, info['text-field']); var hastext = false; if (!text) continue; text = text.toString(); var transform = info['text-transform']; if (transform === 'uppercase') { text = text.toLocaleUpperCase(); } else if (transform === 'lowercase') { text = text.toLocaleLowerCase(); } for (var j = 0, jl = text.length; j < jl; j++) { if (text.charCodeAt(j) <= 65533) { codepoints.push(text.charCodeAt(j)); hastext = true; } } // Track indexes of features with text. if (hastext) { textFeatures[i] = text; } } // get a list of unique codepoints we are missing codepoints = uniq(codepoints, glyphs); return { textFeatures: textFeatures, codepoints: codepoints }; } function uniq(ids, alreadyHave) { var u = []; var last; ids.sort(sortNumbers); for (var i = 0; i < ids.length; i++) { if (ids[i] !== last) { last = ids[i]; if (!alreadyHave[last]) u.push(ids[i]); } } return u; } function sortNumbers(a, b) { return a - b; } },{"../util/token.js":85}],65:[function(require,module,exports){ 'use strict'; var util = require('../util/util.js'), Point = require('point-geometry'); module.exports = { rotationRange: rotationRange, mergeCollisions: mergeCollisions, rotatingFixedCollisions: rotatingFixedCollisions, rotatingRotatingCollisions: rotatingRotatingCollisions, cornerBoxCollisions: cornerBoxCollisions, circleEdgeCollisions: circleEdgeCollisions, getCorners: getCorners, }; /* * Calculate the range a box conflicts with a second box */ function rotationRange(inserting, blocker, scale) { var collisions, box; var a = inserting; var b = blocker; // Instead of scaling the boxes, we move the anchors var relativeAnchor = new Point( (b.anchor.x - a.anchor.x) * scale, (b.anchor.y - a.anchor.y) * scale); // Generate a list of collision interval if (a.hBox && b.hBox) { collisions = rotatingRotatingCollisions(a.box, b.box, relativeAnchor); } else if (a.hBox) { box = { x1: b.box.x1 + relativeAnchor.x, y1: b.box.y1 + relativeAnchor.y, x2: b.box.x2 + relativeAnchor.x, y2: b.box.y2 + relativeAnchor.y }; collisions = rotatingFixedCollisions(a.box, box); } else if (b.hBox) { box = { x1: a.box.x1 - relativeAnchor.x, y1: a.box.y1 - relativeAnchor.y, x2: a.box.x2 - relativeAnchor.x, y2: a.box.y2 - relativeAnchor.y }; collisions = rotatingFixedCollisions(b.box, box); } else { collisions = []; } // Find and return the continous are around 0 where there are no collisions return mergeCollisions(collisions, blocker.placementRange); } /* * Combine an array of collision ranges to form a continuous * range that includes 0. Collisions within the ignoreRange are ignored */ function mergeCollisions(collisions, ignoreRange) { // find continuous interval including 0 that doesn't have any collisions var min = 2 * Math.PI; var max = 0; for (var i = 0; i < collisions.length; i++) { var collision = collisions[i]; var entryOutside = ignoreRange[0] <= collision[0] && collision[0] <= ignoreRange[1]; var exitOutside = ignoreRange[0] <= collision[1] && collision[1] <= ignoreRange[1]; if (entryOutside && exitOutside) { // no collision, since blocker is out of range } else if (entryOutside) { min = Math.min(min, ignoreRange[1]); max = Math.max(max, collision[1]); } else if (exitOutside) { min = Math.min(min, collision[0]); max = Math.max(max, ignoreRange[0]); } else { min = Math.min(min, collision[0]); max = Math.max(max, collision[1]); } } return [min, max]; } /* * Calculate collision ranges for two rotating boxes. */ var horizontal = new Point(1, 0); function rotatingRotatingCollisions(a, b, anchorToAnchor) { var d = anchorToAnchor.mag(); var angleBetweenAnchors = anchorToAnchor.angleWith(horizontal); var c = [], collisions = [], k; // Calculate angles at which collisions may occur // top/bottom c[0] = Math.asin((a.y2 - b.y1) / d); c[1] = Math.asin((a.y2 - b.y1) / d) + Math.PI; c[2] = 2 * Math.PI - Math.asin((-a.y1 + b.y2) / d); c[3] = Math.PI - Math.asin((-a.y1 + b.y2) / d); // left/right c[4] = 2 * Math.PI - Math.acos((a.x2 - b.x1) / d); c[5] = Math.acos((a.x2 - b.x1) / d); c[6] = Math.PI - Math.acos((-a.x1 + b.x2) / d); c[7] = Math.PI + Math.acos((-a.x1 + b.x2) / d); var rl = a.x2 - b.x1; var lr = -a.x1 + b.x2; var tb = a.y2 - b.y1; var bt = -a.y1 + b.y2; // Calculate the distance squared of the diagonal which will be used // to check if the boxes are close enough for collisions to occur at each angle // todo, triple check these var e = []; // top/bottom e[0] = rl * rl + tb * tb; e[1] = lr * lr + tb * tb; e[2] = rl * rl + bt * bt; e[3] = lr * lr + bt * bt; // left/right e[4] = rl * rl + tb * tb; e[5] = rl * rl + bt * bt; e[6] = lr * lr + bt * bt; e[7] = lr * lr + tb * tb; c = c.filter(function(x, i) { // Check if they are close enough to collide return !isNaN(x) && d * d <= e[i]; }).map(function(x) { // So far, angles have been calulated as relative to the vector between anchors. // Convert the angles to angles from north. return (x + angleBetweenAnchors + 2 * Math.PI) % (2 * Math.PI); }); // Group the collision angles by two // each group represents a range where the two boxes collide c.sort(); for (k = 0; k < c.length; k+=2) { collisions.push([c[k], c[k+1]]); } return collisions; } /* * Calculate collision ranges for a rotating box and a fixed box; */ function rotatingFixedCollisions(rotating, fixed) { var cornersR = getCorners(rotating); var cornersF = getCorners(fixed); // A collision occurs when, and only at least one corner from one of the boxes // is within the other box. Calculate these ranges for each corner. var collisions = []; for (var i = 0; i < 4; i++ ) { cornerBoxCollisions(collisions, cornersR[i], cornersF); cornerBoxCollisions(collisions, cornersF[i], cornersR, true); } return collisions; } /* * Calculate the ranges for which the corner, * rotatated around the anchor, is within the box; */ function cornerBoxCollisions(collisions, corner, boxCorners, flip) { var radius = corner.mag(), angles = []; // Calculate the points at which the corners intersect with the edges for (var i = 0, j = 3; i < 4; j = i++) { circleEdgeCollisions(angles, corner, radius, boxCorners[j], boxCorners[i]); } if (angles.length % 2 !== 0) { // TODO fix // This could get hit when a point intersects very close to a corner // and floating point issues cause only one of the entry or exit to be counted throw('expecting an even number of intersections'); } angles.sort(); // Group by pairs, where each represents a range where a collision occurs for (var k = 0; k < angles.length; k+=2) { collisions[k/2] = flip ? [2 * Math.PI - angles[k+1], 2 * Math.PI - angles[k]] : // reflect an angle around 0 degrees [angles[k], angles[k+1]]; } return collisions; } /* * Return the intersection points of a circle and a line segment; */ function circleEdgeCollisions(angles, corner, radius, p1, p2) { var edgeX = p2.x - p1.x; var edgeY = p2.y - p1.y; var a = edgeX * edgeX + edgeY * edgeY; var b = (edgeX * p1.x + edgeY * p1.y) * 2; var c = p1.x * p1.x + p1.y * p1.y - radius * radius; var discriminant = b*b - 4*a*c; // a collision exists only if line intersects circle at two points if (discriminant > 0) { var x1 = (-b - Math.sqrt(discriminant)) / (2 * a); var x2 = (-b + Math.sqrt(discriminant)) / (2 * a); // only add points if within line segment // hack to handle floating point representations of 0 and 1 if (0 < x1 && x1 < 1) { angles.push(getAngle(p1, p2, x1, corner)); } if (0 < x2 && x2 < 1) { angles.push(getAngle(p1, p2, x2, corner)); } } return angles; } function getAngle(p1, p2, d, corner) { return (-corner.angleWithSep( util.interp(p1.x, p2.x, d), util.interp(p1.y, p2.y, d)) + 2 * Math.PI) % (2 * Math.PI); } function getCorners(a) { return [ new Point(a.x1, a.y1), new Point(a.x1, a.y2), new Point(a.x2, a.y2), new Point(a.x2, a.y1) ]; } },{"../util/util.js":87,"point-geometry":98}],66:[function(require,module,exports){ 'use strict'; module.exports = { shape: shape }; function shape(text, name, stacks, maxWidth, lineHeight, horizontalAlign, verticalAlign, justify, spacing, translate) { var glyphs = stacks[name].glyphs; var glyph; var shaping = []; var x = translate[0]; var y = translate[1]; var id; for (var i = 0; i < text.length; i++) { id = text.charCodeAt(i); glyph = glyphs[id]; if (id === 0 || !glyph) continue; shaping.push({ fontstack: name, glyph: id, x: x, y: y }); x += glyph.advance + spacing; } if (!shaping.length) return false; shaping = linewrap(shaping, glyphs, lineHeight, maxWidth, horizontalAlign, verticalAlign, justify); return shaping; } var breakable = { 32: true }; // Currently only breaks at regular spaces function linewrap(shaping, glyphs, lineHeight, maxWidth, horizontalAlign, verticalAlign, justify) { var lastSafeBreak = null; var lengthBeforeCurrentLine = 0; var lineStartIndex = 0; var line = 0; var maxLineLength = 0; if (maxWidth) { for (var i = 0; i < shaping.length; i++) { var shape = shaping[i]; shape.x -= lengthBeforeCurrentLine; shape.y += lineHeight * line; if (shape.x > maxWidth && lastSafeBreak !== null) { var lineLength = shaping[lastSafeBreak + 1].x; maxLineLength = Math.max(lineLength, maxLineLength); for (var k = lastSafeBreak + 1; k <= i; k++) { shaping[k].y += lineHeight; shaping[k].x -= lineLength; } if (justify) { justifyLine(shaping, glyphs, lineStartIndex, lastSafeBreak - 1, justify); } lineStartIndex = lastSafeBreak + 1; lastSafeBreak = null; lengthBeforeCurrentLine += lineLength; line++; } if (breakable[shape.glyph]) { lastSafeBreak = i; } } } maxLineLength = maxLineLength || shaping[shaping.length - 1].x; justifyLine(shaping, glyphs, lineStartIndex, shaping.length - 1, justify); align(shaping, justify, horizontalAlign, verticalAlign, maxLineLength, lineHeight, line); return shaping; } function justifyLine(shaping, glyphs, start, end, justify) { var lastAdvance = glyphs[shaping[end].glyph].advance; var lineIndent = (shaping[end].x + lastAdvance) * justify; for (var j = start; j <= end; j++) { shaping[j].x -= lineIndent; } } function align(shaping, justify, horizontalAlign, verticalAlign, maxLineLength, lineHeight, line) { var shiftX = (justify - horizontalAlign) * maxLineLength; var shiftY = (-verticalAlign * (line + 1) + 0.5) * lineHeight; for (var j = 0; j < shaping.length; j++) { shaping[j].x += shiftX; shaping[j].y += shiftY; } } },{}],67:[function(require,module,exports){ 'use strict'; var Control = require('./control.js'), DOM = require('../../util/dom.js'), util = require('../../util/util.js'); module.exports = Attribution; function Attribution() {} Attribution.prototype = util.inherit(Control, { onAdd: function(map) { var className = 'mapboxgl-ctrl-attrib', container = this._container = DOM.create('div', className, map.container); this._update(); map.on('source.add', this._update.bind(this)); map.on('moveend', this._updateEditLink.bind(this)); return container; }, _update: function() { var attributions = []; for (var id in this._map.sources) { var source = this._map.sources[id]; if (source.tileJSON && source.tileJSON.attribution) { attributions.push(source.tileJSON.attribution); } } this._container.innerHTML = attributions.join(' | '); this._editLink = this._container.getElementsByClassName('mapbox-improve-map')[0]; this._updateEditLink(); }, _updateEditLink: function() { if (this._editLink) { var center = this._map.getCenter(); this._editLink.href = 'https://www.mapbox.com/map-feedback/#/' + center.lng + '/' + center.lat + '/' + Math.round(this._map.getZoom() + 1); } } }); },{"../../util/dom.js":81,"../../util/util.js":87,"./control.js":68}],68:[function(require,module,exports){ 'use strict'; module.exports = Control; function Control() {} Control.prototype = { addTo: function(map) { this._map = map; this._container = this.onAdd(map); return this; }, remove: function () { this._container.parentNode.removeChild(this._container); if (this.onRemove) this.onRemove(this._map); this._map = null; return this; } }; },{}],69:[function(require,module,exports){ 'use strict'; var Control = require('./control.js'), DOM = require('../../util/dom.js'), util = require('../../util/util.js'); module.exports = Navigation; function Navigation() {} Navigation.prototype = util.inherit(Control, { onAdd: function(map) { var className = 'mapboxgl-ctrl-nav'; var container = this._container = DOM.create('div', className, map.container); this._zoomInButton = this._createButton(className + '-zoom-in', map.zoomIn.bind(map)); this._zoomOutButton = this._createButton(className + '-zoom-out', map.zoomOut.bind(map)); this._compass = this._createButton(className + '-compass', map.resetNorth.bind(map)); var compassCanvas = this._compassCanvas = DOM.create('canvas', className + '-compass-canvas', this._compass); compassCanvas.style.cssText = 'width:26px; height:26px;'; compassCanvas.width = 26 * 2; compassCanvas.height = 26 * 2; this._compass.addEventListener('mousedown', this._onCompassDown.bind(this)); this._onCompassMove = this._onCompassMove.bind(this); this._onCompassUp = this._onCompassUp.bind(this); this._compassCtx = compassCanvas.getContext('2d'); map.on('rotate', this._drawNorth.bind(this)); this._drawNorth(); return container; }, _onCompassDown: function(e) { DOM.disableDrag(); document.addEventListener('mousemove', this._onCompassMove); document.addEventListener('mouseup', this._onCompassUp); this._prevX = e.screenX; e.stopPropagation(); }, _onCompassMove: function(e) { var x = e.screenX, d = x < 2 ? -5 : // left edge of the screen, continue rotating x > window.screen.width - 2 ? 5 : // right edge (x - this._prevX) / 4; this._map.setBearing(this._map.getBearing() - d); this._prevX = e.screenX; e.preventDefault(); }, _onCompassUp: function() { document.removeEventListener('mousemove', this._onCompassMove); document.removeEventListener('mouseup', this._onCompassUp); DOM.enableDrag(); }, _createButton: function(className, fn) { var a = DOM.create('a', className, this._container); a.href = '#'; a.addEventListener('click', function (e) { fn(); e.preventDefault(); e.stopPropagation(); }); return a; }, _drawNorth: function() { var rad = 20, width = 8, center = 26, angle = this._map.transform.angle + (Math.PI / 2), ctx = this._compassCtx; this._compassCanvas.width = this._compassCanvas.width; ctx.translate(center, center); ctx.rotate(angle); ctx.beginPath(); ctx.fillStyle = '#000'; ctx.lineTo(0, -width); ctx.lineTo(-rad, 0); ctx.lineTo(0, width); ctx.fill(); ctx.beginPath(); ctx.fillStyle = '#bbb'; ctx.moveTo(0, 0); ctx.lineTo(0, width); ctx.lineTo(rad, 0); ctx.lineTo(0, -width); ctx.fill(); ctx.beginPath(); ctx.strokeStyle = '#fff'; ctx.lineWidth = 4; ctx.moveTo(0, -width); ctx.lineTo(0, width); ctx.stroke(); } }); },{"../../util/dom.js":81,"../../util/util.js":87,"./control.js":68}],70:[function(require,module,exports){ 'use strict'; var util = require('../util/util.js'), browser = require('../util/browser.js'), LatLng = require('../geo/latlng.js'), LatLngBounds = require('../geo/latlngbounds.js'), Point = require('point-geometry'); util.extend(exports, { isEasing: function () { return !!this._stopFn; }, stop: function () { if (this._stopFn) { this._stopFn(); delete this._stopFn; } return this; }, panBy: function(offset, options) { this.panTo(this.transform.center, util.extend({offset: Point.convert(offset).mult(-1)}, options)); return this; }, panTo: function(latlng, options) { this.stop(); latlng = LatLng.convert(latlng); options = util.extend({ duration: 500, easing: util.ease, offset: [0, 0] }, options); var tr = this.transform, offset = Point.convert(options.offset).rotate(-tr.angle), from = tr.point, to = tr.project(latlng).sub(offset); if (!options.noMoveStart) { this.fire('movestart'); } this._stopFn = browser.timed(function(t) { tr.center = tr.unproject(from.add(to.sub(from).mult(options.easing(t)))); this._move(); if (t === 1) this.fire('moveend'); }, options.animate === false ? 0 : options.duration, this); return this; }, // Zooms to a certain zoom level with easing. zoomTo: function(zoom, options) { this.stop(); options = util.extend({ duration: 500 }, options); var tr = this.transform, around = tr.center, easing = this._updateEasing(options.duration, zoom, options.easing), startZoom = tr.zoom; if (options.around) { around = LatLng.convert(options.around); } else if (options.offset) { around = tr.pointLocation(tr.centerPoint.add(Point.convert(options.offset))); } if (options.animate === false) options.duration = 0; if (!this.zooming) { this.zooming = true; this.fire('movestart'); } this._stopFn = browser.timed(function(t) { tr.setZoomAround(util.interp(startZoom, zoom, easing(t)), around); this.style.animationLoop.set(300); // text fading this._move(true); if (t === 1) { this.ease = null; if (options.duration >= 200) { this.fire('moveend'); this.zooming = false; } } }, options.duration, this); if (options.duration < 200) { clearTimeout(this._onZoomEnd); this._onZoomEnd = setTimeout(function() { this.zooming = false; this._rerender(); this.fire('moveend'); }.bind(this), 200); } return this; }, zoomIn: function(options) { this.zoomTo(this.getZoom() + 1, options); }, zoomOut: function(options) { this.zoomTo(this.getZoom() - 1, options); }, rotateTo: function(bearing, options) { this.stop(); options = util.extend({ duration: 500, easing: util.ease }, options); var tr = this.transform, start = this.getBearing(), around = tr.center; if (options.around) { around = LatLng.convert(options.around); } else if (options.offset) { around = tr.pointLocation(tr.centerPoint.add(Point.convert(options.offset))); } this.rotating = true; this.fire('movestart'); this._stopFn = browser.timed(function(t) { if (t === 1) { this.rotating = false; } tr.setBearingAround(util.interp(start, bearing, options.easing(t)), around); this._move(false, true).fire('moveend'); }, options.animate === false ? 0 : options.duration, this); return this; }, resetNorth: function(options) { return this.rotateTo(0, util.extend({duration: 1000}, options)); }, fitBounds: function(bounds, options) { options = util.extend({ padding: 0, offset: [0, 0], maxZoom: Infinity }, options); bounds = LatLngBounds.convert(bounds); var offset = Point.convert(options.offset), tr = this.transform, nw = tr.project(bounds.getNorthWest()), se = tr.project(bounds.getSouthEast()), size = se.sub(nw), center = tr.unproject(nw.add(se).div(2)), scaleX = (tr.width - options.padding * 2 - Math.abs(offset.x) * 2) / size.x, scaleY = (tr.height - options.padding * 2 - Math.abs(offset.y) * 2) / size.y, zoom = Math.min(tr.scaleZoom(tr.scale * Math.min(scaleX, scaleY)), options.maxZoom); return options.linear ? this.easeTo(center, zoom, 0, options) : this.flyTo(center, zoom, 0, options); }, easeTo: function(latlng, zoom, bearing, options) { options = util.extend({ offset: [0, 0], duration: 500, easing: util.ease }, options); latlng = LatLng.convert(latlng); var offset = Point.convert(options.offset), tr = this.transform, startZoom = this.getZoom(), startBearing = this.getBearing(); zoom = zoom === undefined ? startZoom : zoom; bearing = bearing === undefined ? startBearing : bearing; var scale = tr.zoomScale(zoom - startZoom), from = tr.point, to = tr.project(latlng).sub(offset.div(scale)), around = tr.pointLocation(tr.centerPoint.add(to.sub(from).div(1 - 1 / scale))); if (zoom !== startZoom) this.zooming = true; if (startBearing !== bearing) this.rotating = true; this.fire('movestart'); this._stopFn = browser.timed(function (t) { var k = options.easing(t); if (zoom !== startZoom) { tr.setZoomAround(startZoom + k * (zoom - startZoom), around); } if (bearing !== startBearing) { tr.bearing = util.interp(startBearing, bearing, k); } this.style.animationLoop.set(300); // text fading this._move(zoom !== startZoom, bearing !== startBearing); if (t === 1) { this.zooming = false; this.rotating = false; this.fire('moveend'); } }, options.animate === false ? 0 : options.duration, this); return this; }, flyTo: function(latlng, zoom, bearing, options) { options = util.extend({ offset: [0, 0], speed: 1.2, curve: 1.42, easing: util.ease }, options); latlng = LatLng.convert(latlng); var offset = Point.convert(options.offset), tr = this.transform, startZoom = this.getZoom(), startBearing = this.getBearing(); zoom = zoom === undefined ? startZoom : zoom; bearing = bearing === undefined ? startBearing : bearing; var scale = tr.zoomScale(zoom - startZoom), from = tr.point, to = tr.project(latlng).sub(offset.div(scale)); if (options.animate === false) { return this.setView(latlng, zoom, bearing); } var startWorldSize = tr.worldSize, rho = options.curve, V = options.speed, w0 = Math.max(tr.width, tr.height), w1 = w0 / scale, u1 = to.sub(from).mag(), rho2 = rho * rho; function r(i) { var b = (w1 * w1 - w0 * w0 + (i ? -1 : 1) * rho2 * rho2 * u1 * u1) / (2 * (i ? w1 : w0) * rho2 * u1); return Math.log(Math.sqrt(b * b + 1) - b); } function sinh(n) { return (Math.exp(n) - Math.exp(-n)) / 2; } function cosh(n) { return (Math.exp(n) + Math.exp(-n)) / 2; } function tanh(n) { return sinh(n) / cosh(n); } var r0 = r(0), w = function (s) { return (cosh(r0) / cosh(r0 + rho * s)); }, u = function (s) { return w0 * ((cosh(r0) * tanh(r0 + rho * s) - sinh(r0)) / rho2) / u1; }, S = (r(1) - r0) / rho; if (Math.abs(u1) < 0.000001) { if (Math.abs(w0 - w1) < 0.000001) return this; var k = w1 < w0 ? -1 : 1; S = Math.abs(Math.log(w1 / w0)) / rho; u = function() { return 0; }; w = function(s) { return Math.exp(k * rho * s); }; } var duration = 1000 * S / V; this.zooming = true; if (startBearing != bearing) this.rotating = true; this.fire('movestart'); this._stopFn = browser.timed(function (t) { var k = options.easing(t), s = k * S, us = u(s); tr.zoom = startZoom + tr.scaleZoom(1 / w(s)); tr.center = tr.unproject(from.add(to.sub(from).mult(us)), startWorldSize); if (bearing !== startBearing) { tr.bearing = util.interp(startBearing, bearing, k); } this.style.animationLoop.set(300); // text fading this._move(true, bearing !== startBearing); if (t === 1) { this.zooming = false; this.rotating = false; this.fire('moveend'); } }, duration, this); return this; }, _updateEasing: function(duration, zoom, bezier) { var easing; if (this.ease) { var ease = this.ease, t = (Date.now() - ease.start) / ease.duration, speed = ease.easing(t + 0.01) - ease.easing(t), // Quick hack to make new bezier that is continuous with last x = 0.27 / Math.sqrt(speed * speed + 0.0001) * 0.01, y = Math.sqrt(0.27 * 0.27 - x * x); easing = util.bezier(x, y, 0.25, 1); } else { easing = bezier ? util.bezier.apply(util, bezier) : util.ease; } // store information on current easing this.ease = { start: (new Date()).getTime(), to: Math.pow(2, zoom), duration: duration, easing: easing }; return easing; } }); },{"../geo/latlng.js":17,"../geo/latlngbounds.js":18,"../util/browser.js":77,"../util/util.js":87,"point-geometry":98}],71:[function(require,module,exports){ 'use strict'; var Interaction = require('./interaction.js'); var Point = require('point-geometry'); var util = require('../util/util.js'); module.exports = Handlers; function Handlers(map) { var rotateEnd; var inertiaLinearity = 0.2, inertiaEasing = util.bezier(0, 0, inertiaLinearity, 1); this.interaction = new Interaction(map.container) .on('click', function(e) { map.fire('click', e); }) .on('hover', function(e) { map.fire('hover', e); }) .on('down', function () { map.fire('movestart'); }) .on('resize', function() { map.stop(); map.resize(); map.update(); }) .on('pan', function(e) { map.stop(); map.transform.panBy(e.offset); map._move(); }) .on('panend', function(e) { if (!e.inertia) map.fire('moveend'); else { // convert velocity to px/s & adjust for increased initial animation speed when easing out var velocity = e.inertia.mult(1000 * inertiaLinearity), speed = velocity.mag(); var maxSpeed = 4000; // px/s if (speed >= maxSpeed) { speed = maxSpeed; velocity._unit()._mult(maxSpeed); } var deceleration = 8000, // px/s^2 duration = speed / (deceleration * inertiaLinearity), offset = velocity.mult(-duration / 2).round(); map.panBy(offset, { duration: duration * 1000, easing: inertiaEasing, noMoveStart: true }); } }) .on('zoom', function(e) { // Scale by sigmoid of scroll wheel delta. var scale = 2 / (1 + Math.exp(-Math.abs(e.delta / 100))); if (e.delta < 0 && scale !== 0) scale = 1 / scale; var fromScale = map.ease && isFinite(e.delta) ? map.ease.to : map.transform.scale, duration = !isFinite(e.delta) ? 800 : e.source == 'trackpad' ? 0 : 300; map.zoomTo(map.transform.scaleZoom(fromScale * scale), { duration: duration, around: map.unproject(e.point) }); }) .on('rotate', function(e) { var center = map.transform.centerPoint, // Center of rotation startToCenter = e.start.sub(center), startToCenterDist = startToCenter.mag(); // If the first click was too close to the center, move the center of rotation by 200 pixels // in the direction of the click. if (startToCenterDist < 200) { center = e.start.add(new Point(-200, 0)._rotate(startToCenter.angle())); } var bearingDiff = e.prev.sub(center).angleWith(e.current.sub(center)) / Math.PI * 180; map.transform.bearing = map.getBearing() - bearingDiff; map._move(false, true); window.clearTimeout(rotateEnd); rotateEnd = window.setTimeout(function() { map.rotating = false; map._rerender(); }, 200); }); } },{"../util/util.js":87,"./interaction.js":73,"point-geometry":98}],72:[function(require,module,exports){ 'use strict'; module.exports = Hash; var util = require('../util/util.js'); function Hash(map) { this.map = map; window.addEventListener('hashchange', this.onhash.bind(this), false); map.on('move', util.debounce(this.updateHash.bind(this), 100)); } Hash.prototype = { onhash: function() { var loc = location.hash.replace('#', '').split('/'); if (loc.length >= 3) { this.map.setView([+loc[1], +loc[2]], +loc[0], +(loc[3] || 0)); return true; } return false; }, updateHash: function() { var center = this.map.getCenter(), zoom = this.map.getZoom(), bearing = this.map.getBearing(), precision = Math.max(0, Math.ceil(Math.log(zoom) / Math.LN2)), hash = '#' + (Math.round(zoom * 100) / 100) + '/' + center.lat.toFixed(precision) + '/' + center.lng.toFixed(precision) + (bearing ? '/' + (Math.round(bearing * 10) / 10) : ''); window.history.replaceState('', '', hash); } }; },{"../util/util.js":87}],73:[function(require,module,exports){ 'use strict'; var Evented = require('../util/evented.js'), browser = require('../util/browser.js'), Point = require('point-geometry'); module.exports = Interaction; function Interaction(el) { var interaction = this; if (!el) return; var rotating = false, panned = false, firstPos = null, pos = null, inertia = null, now; function mousePos(e) { var rect = el.getBoundingClientRect(); return new Point( e.clientX - rect.left - el.clientLeft, e.clientY - rect.top - el.clientTop); } el.addEventListener('contextmenu', function(ev) { rotating = true; firstPos = pos = mousePos(ev); ev.preventDefault(); }, false); el.addEventListener('mousedown', onmousedown, false); document.addEventListener('mouseup', onmouseup, false); document.addEventListener('mousemove', onmousemove, false); el.addEventListener('click', onclick, false); scrollwheel(zoom); el.addEventListener('dblclick', ondoubleclick, false); window.addEventListener('resize', resize, false); function zoom(type, delta, point) { interaction.fire('zoom', { source: type, delta: delta, point: point }); inertia = null; now = null; } function click(point) { interaction.fire('click', {point: point}); } function hover(point) { interaction.fire('hover', {point: point}); } function pan(point) { if (pos) { var offset = pos.sub(point); interaction.fire('pan', {offset: offset}); // add an averaged version of this movement to the inertia vector if (inertia) { var duration = Date.now() - now; // sometimes it's 0 after some erratic paning if (duration) { var time = duration + now; inertia.push([time, point]); while (inertia.length > 2 && time - inertia[0][0] > 100) inertia.shift(); } } else { inertia = []; } now = Date.now(); pos = point; } } function resize() { interaction.fire('resize'); } function rotate(point) { if (pos) { interaction.fire('rotate', { start: firstPos, prev: pos, current: point }); pos = point; } } function onmousedown(ev) { firstPos = pos = mousePos(ev); interaction.fire('down'); } function onmouseup() { panned = pos && firstPos && (pos.x != firstPos.x || pos.y != firstPos.y); rotating = false; pos = null; if (inertia && inertia.length >= 2 && now > Date.now() - 100) { var last = inertia[inertia.length - 1], first = inertia[0], velocity = last[1].sub(first[1]).div(last[0] - first[0]); interaction.fire('panend', {inertia: velocity}); } else interaction.fire('panend'); inertia = null; now = null; } function onmousemove(ev) { var point = mousePos(ev); if (rotating) { rotate(point); } else if (pos) pan(point); else { var target = ev.toElement; while (target && target != el && target.parentNode) target = target.parentNode; if (target == el) { hover(point); } } } function onclick(ev) { if (!panned) click(mousePos(ev)); } function ondoubleclick(ev) { zoom('wheel', Infinity * (ev.shiftKey ? -1 : 1), mousePos(ev)); ev.preventDefault(); } function scrollwheel(callback) { var firefox = /Firefox/i.test(navigator.userAgent); var safari = /Safari/i.test(navigator.userAgent) && !/Chrom(ium|e)/i.test(navigator.userAgent); var time = window.performance || Date; el.addEventListener('wheel', wheel, false); el.addEventListener('mousewheel', mousewheel, false); var lastEvent = 0; var type = null; var typeTimeout = null; var initialValue = null; function scroll(value, ev) { var stamp = time.now(); var timeDelta = stamp - lastEvent; lastEvent = stamp; var point = mousePos(ev); if (value !== 0 && (value % 4.000244140625) === 0) { // This one is definitely a mouse wheel event. type = 'wheel'; } else if (value !== 0 && Math.abs(value) < 4) { // This one is definitely a trackpad event because it is so small. type = 'trackpad'; } else if (timeDelta > 400) { // This is likely a new scroll action. type = null; initialValue = value; // Start a timeout in case this was a singular event, and dely it // by up to 40ms. typeTimeout = setTimeout(function() { type = 'wheel'; callback(type, -initialValue, point); }, 40); } else if (type === null) { // This is a repeating event, but we don't know the type of event // just yet. If the delta per time is small, we assume it's a // fast trackpad; otherwise we switch into wheel mode. type = (Math.abs(timeDelta * value) < 200) ? 'trackpad' : 'wheel'; // Make sure our delayed event isn't fired again, because we // accumulate the previous event (which was less than 40ms ago) into // this event. if (typeTimeout) { clearTimeout(typeTimeout); typeTimeout = null; value += initialValue; } } // Only fire the callback if we actually know what type of scrolling // device the user uses. if (type !== null) { callback(type, -value, point); } } function wheel(e) { var deltaY = e.deltaY; // Firefox doubles the values on retina screens... if (firefox && e.deltaMode == window.WheelEvent.DOM_DELTA_PIXEL) deltaY /= browser.devicePixelRatio; if (e.deltaMode == window.WheelEvent.DOM_DELTA_LINE) deltaY *= 40; scroll(deltaY, e); e.preventDefault(); } function mousewheel(e) { var deltaY = -e.wheelDeltaY; if (safari) deltaY = deltaY / 3; scroll(deltaY, e); e.preventDefault(); } } } Interaction.prototype = Object.create(Evented); },{"../util/browser.js":77,"../util/evented.js":82,"point-geometry":98}],74:[function(require,module,exports){ 'use strict'; var Dispatcher = require('../util/dispatcher.js'), Canvas = require('../util/canvas.js'), util = require('../util/util.js'), browser = require('../util/browser.js'), ajax = require('../util/ajax.js'), Evented = require('../util/evented.js'), Style = require('../style/style.js'), AnimationLoop = require('../style/animationloop.js'), GLPainter = require('../render/painter.js'), Transform = require('../geo/transform.js'), Hash = require('./hash.js'), Handlers = require('./handlers.js'), Source = require('../source/source.js'), Easings = require('./easings.js'), LatLng = require('../geo/latlng.js'), LatLngBounds = require('../geo/latlngbounds.js'), Point = require('point-geometry'), GlyphSource = require('../symbol/glyphsource.js'), Attribution = require('./control/attribution.js'); // allow redefining Map here (jshint thinks it's global) // jshint -W079 var Map = module.exports = function(options) { options = this.options = util.inherit(this.options, options); this.animationLoop = new AnimationLoop(); this.transform = new Transform(options.minZoom, options.maxZoom); this.hash = options.hash && new Hash(this); if (options.maxBounds) { var b = LatLngBounds.convert(options.maxBounds); this.transform.latRange = [b.getSouth(), b.getNorth()]; this.transform.lngRange = [b.getWest(), b.getEast()]; } this._onStyleChange = this._onStyleChange.bind(this); this._updateBuckets = this._updateBuckets.bind(this); this.render = this.render.bind(this); this._setupContainer(); this._setupPainter(); this.handlers = options.interactive && new Handlers(this); this.dispatcher = new Dispatcher(Math.max(options.numWorkers, 1), this); // don't set position from options if set through hash if (!this.hash || !this.hash.onhash()) { this.setView(options.center, options.zoom, options.bearing); } this.sources = {}; this.stacks = {}; this.resize(); if (typeof options.style === 'object') { this.setStyle(options.style); } else if (typeof options.style === 'string') { ajax.getJSON(options.style, function (err, data) { if (err) throw err; this.setStyle(data); }.bind(this)); } if (options.attributionControl) this.addControl(new Attribution()); }; util.extend(Map.prototype, Evented); util.extend(Map.prototype, Easings); util.extend(Map.prototype, { options: { center: [0, 0], zoom: 0, bearing: 0, minZoom: 0, maxZoom: 20, numWorkers: browser.hardwareConcurrency - 1, interactive: true, hash: false, attributionControl: true }, addSource: function(id, source) { this.sources[id] = source; source.id = id; if (source.onAdd) { source.onAdd(this); } if (source.enabled) source.fire('source.add', {source: source}); return this; }, removeSource: function(id) { var source = this.sources[id]; if (source.onRemove) { source.onRemove(this); } delete this.sources[id]; return this.fire('source.remove', {source: source}); }, addControl: function(control) { control.addTo(this); return this; }, // Set the map's center, zoom, and bearing setView: function(center, zoom, bearing) { this.stop(); var tr = this.transform, zoomChanged = tr.zoom !== +zoom, bearingChanged = tr.bearing !== +bearing; tr.center = LatLng.convert(center); tr.zoom = +zoom; tr.bearing = +bearing; return this .fire('movestart') ._move(zoomChanged, bearingChanged) .fire('moveend'); }, setCenter: function(center) { this.setView(center, this.getZoom(), this.getBearing()); }, setZoom: function(zoom) { this.setView(this.getCenter(), zoom, this.getBearing()); }, setBearing: function(bearing) { this.setView(this.getCenter(), this.getZoom(), bearing); }, getCenter: function() { return this.transform.center; }, getZoom: function() { return this.transform.zoom; }, getBearing: function() { return this.transform.bearing; }, // Detect the map's new width and height and resize it. resize: function() { var width = 0, height = 0; if (this.container) { width = this.container.offsetWidth || 400; height = this.container.offsetHeight || 300; } this.canvas.resize(width, height); this.transform.width = width; this.transform.height = height; this.transform._constrain(); if (this.style && this.style.sprite) { this.style.sprite.resize(this.painter.gl); } this.painter.resize(width, height); return this .fire('movestart') ._move() .fire('resize') .fire('moveend'); }, getBounds: function() { return new LatLngBounds( this.transform.pointLocation(new Point(0, 0)), this.transform.pointLocation(this.transform.size)); }, project: function(latlng) { return this.transform.locationPoint(LatLng.convert(latlng)); }, unproject: function(point) { return this.transform.pointLocation(Point.convert(point)); }, featuresAt: function(point, params, callback) { var features = []; var error = null; var map = this; point = Point.convert(point); util.asyncEach(Object.keys(this.sources), function(id, callback) { var source = map.sources[id]; source.featuresAt(point, params, function(err, result) { if (result) features = features.concat(result); if (err) error = err; callback(); }); }, function() { callback(error, features); }); return this; }, setStyle: function(style) { if (this.style) { this.style.off('change', this._onStyleChange); } if (style instanceof Style) { this.style = style; } else { this.style = new Style(style, this.animationLoop); } var sources = this.style.stylesheet.sources; for (var id in sources) { this.addSource(id, Source.create(sources[id])); } this.glyphSource = new GlyphSource(this.style.stylesheet.glyphs, this.painter.glyphAtlas); this.style.on('change', this._onStyleChange); this._styleDirty = true; this._tilesDirty = true; this._updateBuckets(); this._updateGlyphs(); this.fire('style.change'); return this; }, _move: function (zoom, rotate) { this.update(zoom).fire('move'); if (zoom) this.fire('zoom'); if (rotate) this.fire('rotate'); return this; }, // map setup code _setupContainer: function() { var id = this.options.container; var container = this.container = typeof id === 'string' ? document.getElementById(id) : id; if (container) container.classList.add('mapboxgl-map'); this.canvas = new Canvas(this, container); }, _setupPainter: function() { var gl = this.canvas.getWebGLContext(); if (!gl) { alert('Failed to initialize WebGL'); return; } this.painter = new GLPainter(gl, this.transform); }, _contextLost: function(event) { event.preventDefault(); if (this._frameId) { browser.cancelFrame(this._frameId); } }, _contextRestored: function() { this._setupPainter(); this.resize(); this.update(); }, // Callbacks from web workers 'debug message': function(data) { console.log.apply(console, data); }, 'alert message': function(data) { alert.apply(window, data); }, 'get sprite json': function(params, callback) { var sprite = this.style.sprite; if (sprite.loaded()) { callback(null, { sprite: sprite.data, retina: sprite.retina }); } else { sprite.on('loaded', function() { callback(null, { sprite: sprite.data, retina: sprite.retina }); }); } }, 'get glyphs': function(params, callback) { this.glyphSource.getRects(params.fontstack, params.codepoints, params.id, callback); }, // Rendering update: function(updateStyle) { if (!this.style) return this; this._styleDirty = this._styleDirty || updateStyle; this._tilesDirty = true; this._rerender(); return this; }, // Call when a (re-)render of the map is required, e.g. when the user panned or zoomed,f or new data is available. render: function() { if (this._styleDirty) { this._styleDirty = false; this._updateStyle(); } if (this._tilesDirty) { for (var id in this.sources) { this.sources[id].update(); } this._tilesDirty = false; } this._renderGroups(this.style.layerGroups); this.fire('render'); this._frameId = null; if (!this.animationLoop.stopped()) { this._styleDirty = true; } if (this._repaint || !this.animationLoop.stopped()) { this._rerender(); } return this; }, _renderGroups: function(groups, name) { var i, len, group, source, k; // Render all dependencies (composited layers) to textures for (i = 0, len = groups.length; i < len; i++) { group = groups[i]; for (k in group.dependencies) { this._renderGroups(group.dependencies[k], k); } } // attach render destination. if no name, main canvas. this.painter.bindRenderTexture(name); // Render the groups for (i = 0, len = groups.length; i < len; i++) { group = groups[i]; source = this.sources[group.source]; if (source) { this.painter.clearStencil(); source.render(group); } else if (group.composited) { this.painter.draw(undefined, this.style, group, {}); } else if (group.source === undefined) { this.painter.draw(undefined, this.style, group, { background: true }); } } }, _rerender: function() { if (!this._frameId) { this._frameId = browser.frame(this.render); } }, _onStyleChange: function () { this.update(true); }, _updateStyle: function() { if (!this.style) return; this.style.recalculate(this.transform.zoom); }, _updateGlyphs: function() { this.dispatcher.broadcast('set glyphs', this.style.stylesheet.glyphs); }, _updateBuckets: function() { // Transfer a stripped down version of the style to the workers. They only // need the bucket information to know what features to extract from the tile. this.dispatcher.broadcast('set buckets', this.style.orderedBuckets); // clears all tiles to recalculate geometries (for changes to linecaps, linejoins, ...) for (var s in this.sources) { this.sources[s].load(); } this.update(); } }); util.extendAll(Map.prototype, { // debug code _debug: false, get debug() { return this._debug; }, set debug(value) { this._debug = value; this._rerender(); }, // continuous repaint _repaint: false, get repaint() { return this._repaint; }, set repaint(value) { this._repaint = value; this._rerender(); }, // polygon antialiasing _antialiasing: true, get antialiasing() { return this._antialiasing; }, set antialiasing(value) { this._antialiasing = value; this._rerender(); }, // show vertices _vertices: false, get vertices() { return this._vertices; }, set vertices(value) { this._vertices = value; this._rerender(); }, // show vertices _loadNewTiles: true, get loadNewTiles() { return this._loadNewTiles; }, set loadNewTiles(value) { this._loadNewTiles = value; this.update(); } }); },{"../geo/latlng.js":17,"../geo/latlngbounds.js":18,"../geo/transform.js":19,"../render/painter.js":33,"../source/source.js":40,"../style/animationloop.js":48,"../style/style.js":53,"../symbol/glyphsource.js":61,"../util/ajax.js":76,"../util/browser.js":77,"../util/canvas.js":78,"../util/dispatcher.js":79,"../util/evented.js":82,"../util/util.js":87,"./control/attribution.js":67,"./easings.js":70,"./handlers.js":71,"./hash.js":72,"point-geometry":98}],75:[function(require,module,exports){ 'use strict'; module.exports = Actor; function Actor(target, parent) { this.target = target; this.parent = parent; this.callbacks = {}; this.callbackID = 0; this.receive = this.receive.bind(this); this.target.addEventListener('message', this.receive, false); } Actor.prototype.receive = function(message) { var data = message.data, callback; if (data.type == '') { callback = this.callbacks[data.id]; delete this.callbacks[data.id]; callback(data.error || null, data.data); } else if (typeof data.id !== 'undefined') { var id = data.id; this.parent[data.type](data.data, function response(err, data, buffers) { // console.warn('trying to clone', data, buffers, message.target); message.target.postMessage({ type: '', id: String(id), error: err ? String(err) : null, data: data }, buffers); }); } else { this.parent[data.type](data.data); } }; Actor.prototype.send = function(type, data, callback, buffers) { var id = null; if (callback) this.callbacks[id = this.callbackID++] = callback; this.target.postMessage({ type: type, id: String(id), data: data }, buffers); }; },{}],76:[function(require,module,exports){ 'use strict'; exports.getJSON = function(url, callback) { var xhr = new XMLHttpRequest(); xhr.open('GET', url, true); xhr.onerror = function(e) { callback(e); }; xhr.onload = function() { if (xhr.status >= 200 && xhr.status < 300 && xhr.response) { var data; try { data = JSON.parse(xhr.response); } catch (err) { return callback(err); } callback(null, data); } else { callback(new Error(xhr.statusText)); } }; xhr.send(); return xhr; }; exports.getArrayBuffer = function(url, callback) { var xhr = new XMLHttpRequest(); xhr.open('GET', url, true); xhr.responseType = 'arraybuffer'; xhr.onerror = function(e) { callback(e); }; xhr.onload = function() { if (xhr.status >= 200 && xhr.status < 300 && xhr.response) { callback(null, xhr.response); } else { callback(new Error(xhr.statusText)); } }; xhr.send(); return xhr; }; exports.getImage = function(url, callback) { var img = new Image(); img.crossOrigin = 'Anonymous'; img.onload = function() { callback(null, img); }; img.src = url; img.getData = function() { return getImageData(this); }; return img; }; function getImageData(img) { var canvas = document.createElement('canvas'); var context = canvas.getContext('2d'); canvas.width = img.width; canvas.height = img.height; context.drawImage(img, 0, 0); return context.getImageData(0, 0, img.width, img.height).data; } },{}],77:[function(require,module,exports){ 'use strict'; var frameName = (function() { if (window.requestAnimationFrame) return 'requestAnimationFrame'; if (window.mozRequestAnimationFrame) return 'mozRequestAnimationFrame'; if (window.webkitRequestAnimationFrame) return 'webkitRequestAnimationFrame'; if (window.msRequestAnimationFrame) return 'msRequestAnimationFrame'; })(); exports.frame = function(fn) { return window[frameName](fn); }; exports.cancelFrame = function(id) { (window.cancelRequestAnimationFrame || window.mozCancelRequestAnimationFrame || window.webkitCancelRequestAnimationFrame || window.msCancelRequestAnimationFrame)(id); }; exports.timed = function (fn, dur, ctx) { if (!dur) { return fn.call(ctx, 1); } var abort = false, start = window.performance ? window.performance.now() : Date.now(); function tick(now) { if (abort) return; if (!window.performance) now = Date.now(); if (now > start + dur) { fn.call(ctx, 1); } else { fn.call(ctx, (now - start) / dur); exports.frame(tick); } } exports.frame(tick); return function() { abort = true; }; }; exports.supported = function() { var supports = [ function() { return typeof window !== 'undefined'; }, function() { return typeof document !== 'undefined'; }, function () { return !!(Array.prototype && Array.prototype.every && Array.prototype.filter && Array.prototype.forEach && Array.prototype.indexOf && Array.prototype.lastIndexOf && Array.prototype.map && Array.prototype.some && Array.prototype.reduce && Array.prototype.reduceRight && Array.isArray); }, function() { return !!(Function.prototype && Function.prototype.bind), !!(Object.keys && Object.create && Object.getPrototypeOf && Object.getOwnPropertyNames && Object.isSealed && Object.isFrozen && Object.isExtensible && Object.getOwnPropertyDescriptor && Object.defineProperty && Object.defineProperties && Object.seal && Object.freeze && Object.preventExtensions); }, function() { return 'JSON' in window && 'parse' in JSON && 'stringify' in JSON; }, function() { var canvas = document.createElement('canvas'); if ('supportsContext' in canvas) { return canvas.supportsContext('webgl') || canvas.supportsContext('experimental-webgl'); } return !!window.WebGLRenderingContext && (!!canvas.getContext('webgl') || !!canvas.getContext('experimental-webgl')); }, function() { return 'Worker' in window; } ]; for (var i = 0; i < supports.length; i++) { if (!supports[i]()) return false; } return true; }; exports.hardwareConcurrency = navigator.hardwareConcurrency || 8; Object.defineProperty(exports, 'devicePixelRatio', { get: function() { return window.devicePixelRatio; } }); },{}],78:[function(require,module,exports){ 'use strict'; module.exports = Canvas; function Canvas(parent, container) { this.canvas = document.createElement('canvas'); this.canvas.style.position = 'absolute'; this.canvas.classList.add('mapboxgl-canvas'); this.canvas.addEventListener('webglcontextlost', parent._contextLost.bind(parent), false); this.canvas.addEventListener('webglcontextrestored', parent._contextRestored.bind(parent), false); container.appendChild(this.canvas); } Canvas.prototype.resize = function(width, height) { var pixelRatio = window.devicePixelRatio || 1; // Request the required canvas size taking the pixelratio into account. this.canvas.width = pixelRatio * width; this.canvas.height = pixelRatio * height; // Maintain the same canvas size, potentially downscaling it for HiDPI displays this.canvas.style.width = width + 'px'; this.canvas.style.height = height + 'px'; }; Canvas.prototype.getWebGLContext = function() { return this.canvas.getContext("experimental-webgl", { antialias: false, alpha: true, stencil: true, depth: false }); }; },{}],79:[function(require,module,exports){ 'use strict'; var Actor = require('../actor.js'); var scripts = document.getElementsByTagName("script"); var workerFile = scripts[scripts.length - 1].getAttribute('src'); var absolute = workerFile.indexOf('http') !== -1; // Manages the WebWorkers module.exports = Dispatcher; function Dispatcher(length, parent) { this.actors = []; this.currentActor = 0; var url, blob, i; for (i = 0; i < length; i++) { // due to cross domain issues we can't load it directly with the url, // so create a blob and object url and load that if (absolute) { blob = new Blob(['importScripts("' + workerFile + '");'], {type : 'application/javascript'}); url = window.URL.createObjectURL(blob); } else { url = workerFile; } var worker = new Worker(url); var actor = new Actor(worker, parent); actor.name = "Worker " + i; this.actors.push(actor); } } Dispatcher.prototype.broadcast = function(type, data) { for (var i = 0; i < this.actors.length; i++) { this.actors[i].send(type, data); } }; Dispatcher.prototype.send = function(type, data, callback, targetID, buffers) { if (typeof targetID !== 'number' || isNaN(targetID)) { // Use round robin to send requests to web workers. targetID = this.currentActor = (this.currentActor + 1) % this.actors.length; } this.actors[targetID].send(type, data, callback, buffers); return targetID; }; },{"../actor.js":75}],80:[function(require,module,exports){ 'use strict'; module.exports = { HTTP_URL: 'http://a.tiles.mapbox.com/v4', HTTPS_URL: 'https://a.tiles.mapbox.com/v4', FORCE_HTTPS: false, REQUIRE_ACCESS_TOKEN: true }; },{}],81:[function(require,module,exports){ 'use strict'; exports.create = function (tagName, className, container) { var el = document.createElement(tagName); if (className) el.className = className; if (container) container.appendChild(el); return el; }; function preventDefault(e) { e.preventDefault(); } var docEl = typeof document !== 'undefined' ? document.documentElement : {}, selectProp = 'userSelect' in docEl ? 'userSelect' : 'MozUserSelect' in docEl ? 'MozUserSelect' : 'WebkitUserSelect' in docEl ? 'WebkitUserSelect' : null, userSelect; exports.disableDrag = function () { window.addEventListener('dragstart', preventDefault); if ('onselectstart' in document) window.addEventListener('selectstart', preventDefault); else if (selectProp) { userSelect = docEl.style[selectProp]; docEl.style[selectProp] = 'none'; } }; exports.enableDrag = function () { window.removeEventListener('dragstart', preventDefault); if ('onselectstart' in document) window.removeEventListener('selectstart', preventDefault); else if (selectProp) docEl.style[selectProp] = userSelect; }; },{}],82:[function(require,module,exports){ 'use strict'; var util = require('./util.js'); module.exports = { on: function(type, fn) { this._events = this._events || {}; this._events[type] = this._events[type] || []; this._events[type].push(fn); return this; }, off: function(type, fn) { if (!type) { // clear all listeners if no arguments specified delete this._events; return this; } if (!this.listens(type)) return this; if (fn) { var idx = this._events[type].indexOf(fn); if (idx >= 0) { this._events[type].splice(idx, 1); } if (!this._events[type].length) { delete this._events[type]; } } else { delete this._events[type]; } return this; }, fire: function(type, data) { if (!this.listens(type)) return this; data = util.extend({}, data); util.extend(data, {type: type, target: this}); // make sure adding/removing listeners inside other listeners won't cause infinite loop var listeners = this._events[type].slice(); for (var i = 0; i < listeners.length; i++) { listeners[i].call(this, data); } return this; }, listens: function(type) { return !!(this._events && this._events[type]); } }; },{"./util.js":87}],83:[function(require,module,exports){ 'use strict'; module.exports = Glyphs; function Glyphs(buffer, end) { // Public this.stacks = {}; // Private this._buffer = buffer; var val, tag; if (typeof end === 'undefined') end = buffer.length; while (buffer.pos < end) { val = buffer.readVarint(); tag = val >> 3; if (tag == 1) { var fontstack = this.readFontstack(); this.stacks[fontstack.name] = fontstack; } else { // console.warn('skipping tile tag ' + tag); buffer.skip(val); } } } Glyphs.prototype.readFontstack = function() { var buffer = this._buffer; var fontstack = { glyphs: {} }; var bytes = buffer.readVarint(); var val, tag; var end = buffer.pos + bytes; while (buffer.pos < end) { val = buffer.readVarint(); tag = val >> 3; if (tag == 1) { fontstack.name = buffer.readString(); } else if (tag == 2) { var range = buffer.readString(); fontstack.range = range; } else if (tag == 3) { var glyph = this.readGlyph(); fontstack.glyphs[glyph.id] = glyph; } else { buffer.skip(val); } } return fontstack; }; Glyphs.prototype.readGlyph = function() { var buffer = this._buffer; var glyph = {}; var bytes = buffer.readVarint(); var val, tag; var end = buffer.pos + bytes; while (buffer.pos < end) { val = buffer.readVarint(); tag = val >> 3; if (tag == 1) { glyph.id = buffer.readVarint(); } else if (tag == 2) { glyph.bitmap = buffer.readBuffer(); } else if (tag == 3) { glyph.width = buffer.readVarint(); } else if (tag == 4) { glyph.height = buffer.readVarint(); } else if (tag == 5) { glyph.left = buffer.readSVarint(); } else if (tag == 6) { glyph.top = buffer.readSVarint(); } else if (tag == 7) { glyph.advance = buffer.readVarint(); } else { buffer.skip(val); } } return glyph; }; },{}],84:[function(require,module,exports){ 'use strict'; /* * A [most-recently-used cache](http://en.wikipedia.org/wiki/Cache_algorithms) * with hash lookup made possible by keeping a list of keys in parallel to * an array of dictionary of values */ module.exports = MRUCache; function MRUCache(length, onRemove) { this.max = length; this.onRemove = onRemove; this.reset(); } /* * Clears the cache */ MRUCache.prototype.reset = function() { this.list = {}; this.order = []; return this; }; /* * Add a key, value combination to the cache, trimming its size if this pushes * it over max length. */ MRUCache.prototype.add = function(key, data) { this.list[key] = data; this.order.push(key); if (this.order.length > this.max) { var removedData = this.get(this.order[0]); if (removedData) this.onRemove(removedData); } return this; }; /* * Determine whether the value attached to `key` is present */ MRUCache.prototype.has = function(key) { return key in this.list; }; /* * List all keys in the cache */ MRUCache.prototype.keys = function() { return this.order; }; /* * Get the value attached to a specific key. If the key is not found, * returns `null` */ MRUCache.prototype.get = function(key) { if (!this.has(key)) { return null; } var data = this.list[key]; delete this.list[key]; this.order.splice(this.order.indexOf(key), 1); return data; }; },{}],85:[function(require,module,exports){ 'use strict'; module.exports = resolveTokens; var tokenPattern = /{([\w-]+)}/; function resolveTokens(properties, expression) { var match; var value; var text = expression; while ((match = text.match(tokenPattern))) { value = typeof properties[match[1]] === 'undefined' ? '' : properties[match[1]]; text = text.replace(match[0], value); } return text; } },{}],86:[function(require,module,exports){ 'use strict'; var config = require('./config'); module.exports = function(path, accessToken) { accessToken = accessToken || config.ACCESS_TOKEN; if (!accessToken && config.REQUIRE_ACCESS_TOKEN) { throw new Error('An API access token is required to use Mapbox GL. ' + 'See https://www.mapbox.com/developers/api/#access-tokens'); } var url = ((typeof document !== 'undefined' && 'https:' === document.location.protocol) || config.FORCE_HTTPS) ? config.HTTPS_URL : config.HTTP_URL; url += path; url += url.indexOf('?') !== -1 ? '&access_token=' : '?access_token='; if (config.REQUIRE_ACCESS_TOKEN) { if (accessToken[0] === 's') { throw new Error('Use a public access token (pk.*) with Mapbox GL JS, not a secret access token (sk.*). ' + 'See https://www.mapbox.com/developers/api/#access-tokens'); } url += accessToken; } return url; }; module.exports.tileJSON = function(mapID, accessToken) { var url = module.exports('/' + mapID + '.json', accessToken); // TileJSON requests need a secure flag appended to their URLs so // that the server knows to send SSL-ified resource references. if (url.indexOf('https') === 0) url += '&secure'; return url; }; },{"./config":80}],87:[function(require,module,exports){ 'use strict'; var UnitBezier = require('unitbezier'); exports.easeCubicInOut = function (t) { if (t <= 0) return 0; if (t >= 1) return 1; var t2 = t * t, t3 = t2 * t; return 4 * (t < 0.5 ? t3 : 3 * (t - t2) + t3 - 0.75); }; exports.bezier = function(p1x, p1y, p2x, p2y) { var bezier = new UnitBezier(p1x, p1y, p2x, p2y); return function(t) { return bezier.solve(t); }; }; exports.ease = exports.bezier(0.25, 0.1, 0.25, 1); exports.interp = function (a, b, t) { return (a * (1 - t)) + (b * t); }; exports.premultiply = function (c) { c[0] *= c[3]; c[1] *= c[3]; c[2] *= c[3]; return c; }; exports.asyncEach = function (array, fn, callback) { var remaining = array.length; if (remaining === 0) return callback(); function check() { if (--remaining === 0) callback(); } for (var i = 0; i < array.length; i++) fn(array[i], check); }; exports.keysDifference = function (obj, other) { var difference = []; for (var i in obj) { if (!(i in other)) { difference.push(i); } } return difference; }; exports.extend = function (dest, src) { for (var i in src) { dest[i] = src[i]; } return dest; }; exports.extendAll = function (dest, src) { for (var i in src) { Object.defineProperty(dest, i, Object.getOwnPropertyDescriptor(src, i)); } return dest; }; exports.inherit = function (parent, props) { var parentProto = typeof parent === 'function' ? parent.prototype : parent, proto = Object.create(parentProto); exports.extendAll(proto, props); return proto; }; var id = 1; exports.uniqueId = function () { return id++; }; exports.throttle = function (fn, time, context) { var lock, args, wrapperFn, later; later = function () { // reset lock and call if queued lock = false; if (args) { wrapperFn.apply(context, args); args = false; } }; wrapperFn = function () { if (lock) { // called too soon, queue to call later args = arguments; } else { // call and lock until later fn.apply(context, arguments); setTimeout(later, time); lock = true; } }; return wrapperFn; }; exports.debounce = function(fn, time) { var timer, args; return function() { args = arguments; clearTimeout(timer); timer = setTimeout(function() { fn.apply(null, args); }, time); }; }; },{"unitbezier":101}],88:[function(require,module,exports){ /*! * The buffer module from node.js, for the browser. * * @author Feross Aboukhadijeh * @license MIT */ var base64 = require('base64-js') var ieee754 = require('ieee754') exports.Buffer = Buffer exports.SlowBuffer = Buffer exports.INSPECT_MAX_BYTES = 50 Buffer.poolSize = 8192 /** * If `TYPED_ARRAY_SUPPORT`: * === true Use Uint8Array implementation (fastest) * === false Use Object implementation (most compatible, even IE6) * * Browsers that support typed arrays are IE 10+, Firefox 4+, Chrome 7+, Safari 5.1+, * Opera 11.6+, iOS 4.2+. * * Note: * * - Implementation must support adding new properties to `Uint8Array` instances. * Firefox 4-29 lacked support, fixed in Firefox 30+. * See: https://bugzilla.mozilla.org/show_bug.cgi?id=695438. * * - Chrome 9-10 is missing the `TypedArray.prototype.subarray` function. * * - IE10 has a broken `TypedArray.prototype.subarray` function which returns arrays of * incorrect length in some situations. * * We detect these buggy browsers and set `TYPED_ARRAY_SUPPORT` to `false` so they will * get the Object implementation, which is slower but will work correctly. */ var TYPED_ARRAY_SUPPORT = (function () { try { var buf = new ArrayBuffer(0) var arr = new Uint8Array(buf) arr.foo = function () { return 42 } return 42 === arr.foo() && // typed array instances can be augmented typeof arr.subarray === 'function' && // chrome 9-10 lack `subarray` new Uint8Array(1).subarray(1, 1).byteLength === 0 // ie10 has broken `subarray` } catch (e) { return false } })() /** * Class: Buffer * ============= * * The Buffer constructor returns instances of `Uint8Array` that are augmented * with function properties for all the node `Buffer` API functions. We use * `Uint8Array` so that square bracket notation works as expected -- it returns * a single octet. * * By augmenting the instances, we can avoid modifying the `Uint8Array` * prototype. */ function Buffer (subject, encoding, noZero) { if (!(this instanceof Buffer)) return new Buffer(subject, encoding, noZero) var type = typeof subject // Find the length var length if (type === 'number') length = subject > 0 ? subject >>> 0 : 0 else if (type === 'string') { if (encoding === 'base64') subject = base64clean(subject) length = Buffer.byteLength(subject, encoding) } else if (type === 'object' && subject !== null) { // assume object is array-like if (subject.type === 'Buffer' && isArray(subject.data)) subject = subject.data length = +subject.length > 0 ? Math.floor(+subject.length) : 0 } else throw new Error('First argument needs to be a number, array or string.') var buf if (TYPED_ARRAY_SUPPORT) { // Preferred: Return an augmented `Uint8Array` instance for best performance buf = Buffer._augment(new Uint8Array(length)) } else { // Fallback: Return THIS instance of Buffer (created by `new`) buf = this buf.length = length buf._isBuffer = true } var i if (TYPED_ARRAY_SUPPORT && typeof subject.byteLength === 'number') { // Speed optimization -- use set if we're copying from a typed array buf._set(subject) } else if (isArrayish(subject)) { // Treat array-ish objects as a byte array if (Buffer.isBuffer(subject)) { for (i = 0; i < length; i++) buf[i] = subject.readUInt8(i) } else { for (i = 0; i < length; i++) buf[i] = ((subject[i] % 256) + 256) % 256 } } else if (type === 'string') { buf.write(subject, 0, encoding) } else if (type === 'number' && !TYPED_ARRAY_SUPPORT && !noZero) { for (i = 0; i < length; i++) { buf[i] = 0 } } return buf } // STATIC METHODS // ============== Buffer.isEncoding = function (encoding) { switch (String(encoding).toLowerCase()) { case 'hex': case 'utf8': case 'utf-8': case 'ascii': case 'binary': case 'base64': case 'raw': case 'ucs2': case 'ucs-2': case 'utf16le': case 'utf-16le': return true default: return false } } Buffer.isBuffer = function (b) { return !!(b != null && b._isBuffer) } Buffer.byteLength = function (str, encoding) { var ret str = str.toString() switch (encoding || 'utf8') { case 'hex': ret = str.length / 2 break case 'utf8': case 'utf-8': ret = utf8ToBytes(str).length break case 'ascii': case 'binary': case 'raw': ret = str.length break case 'base64': ret = base64ToBytes(str).length break case 'ucs2': case 'ucs-2': case 'utf16le': case 'utf-16le': ret = str.length * 2 break default: throw new Error('Unknown encoding') } return ret } Buffer.concat = function (list, totalLength) { assert(isArray(list), 'Usage: Buffer.concat(list[, length])') if (list.length === 0) { return new Buffer(0) } else if (list.length === 1) { return list[0] } var i if (totalLength === undefined) { totalLength = 0 for (i = 0; i < list.length; i++) { totalLength += list[i].length } } var buf = new Buffer(totalLength) var pos = 0 for (i = 0; i < list.length; i++) { var item = list[i] item.copy(buf, pos) pos += item.length } return buf } Buffer.compare = function (a, b) { assert(Buffer.isBuffer(a) && Buffer.isBuffer(b), 'Arguments must be Buffers') var x = a.length var y = b.length for (var i = 0, len = Math.min(x, y); i < len && a[i] === b[i]; i++) {} if (i !== len) { x = a[i] y = b[i] } if (x < y) { return -1 } if (y < x) { return 1 } return 0 } // BUFFER INSTANCE METHODS // ======================= function hexWrite (buf, string, offset, length) { offset = Number(offset) || 0 var remaining = buf.length - offset if (!length) { length = remaining } else { length = Number(length) if (length > remaining) { length = remaining } } // must be an even number of digits var strLen = string.length assert(strLen % 2 === 0, 'Invalid hex string') if (length > strLen / 2) { length = strLen / 2 } for (var i = 0; i < length; i++) { var byte = parseInt(string.substr(i * 2, 2), 16) assert(!isNaN(byte), 'Invalid hex string') buf[offset + i] = byte } return i } function utf8Write (buf, string, offset, length) { var charsWritten = blitBuffer(utf8ToBytes(string), buf, offset, length) return charsWritten } function asciiWrite (buf, string, offset, length) { var charsWritten = blitBuffer(asciiToBytes(string), buf, offset, length) return charsWritten } function binaryWrite (buf, string, offset, length) { return asciiWrite(buf, string, offset, length) } function base64Write (buf, string, offset, length) { var charsWritten = blitBuffer(base64ToBytes(string), buf, offset, length) return charsWritten } function utf16leWrite (buf, string, offset, length) { var charsWritten = blitBuffer(utf16leToBytes(string), buf, offset, length) return charsWritten } Buffer.prototype.write = function (string, offset, length, encoding) { // Support both (string, offset, length, encoding) // and the legacy (string, encoding, offset, length) if (isFinite(offset)) { if (!isFinite(length)) { encoding = length length = undefined } } else { // legacy var swap = encoding encoding = offset offset = length length = swap } offset = Number(offset) || 0 var remaining = this.length - offset if (!length) { length = remaining } else { length = Number(length) if (length > remaining) { length = remaining } } encoding = String(encoding || 'utf8').toLowerCase() var ret switch (encoding) { case 'hex': ret = hexWrite(this, string, offset, length) break case 'utf8': case 'utf-8': ret = utf8Write(this, string, offset, length) break case 'ascii': ret = asciiWrite(this, string, offset, length) break case 'binary': ret = binaryWrite(this, string, offset, length) break case 'base64': ret = base64Write(this, string, offset, length) break case 'ucs2': case 'ucs-2': case 'utf16le': case 'utf-16le': ret = utf16leWrite(this, string, offset, length) break default: throw new Error('Unknown encoding') } return ret } Buffer.prototype.toString = function (encoding, start, end) { var self = this encoding = String(encoding || 'utf8').toLowerCase() start = Number(start) || 0 end = (end === undefined) ? self.length : Number(end) // Fastpath empty strings if (end === start) return '' var ret switch (encoding) { case 'hex': ret = hexSlice(self, start, end) break case 'utf8': case 'utf-8': ret = utf8Slice(self, start, end) break case 'ascii': ret = asciiSlice(self, start, end) break case 'binary': ret = binarySlice(self, start, end) break case 'base64': ret = base64Slice(self, start, end) break case 'ucs2': case 'ucs-2': case 'utf16le': case 'utf-16le': ret = utf16leSlice(self, start, end) break default: throw new Error('Unknown encoding') } return ret } Buffer.prototype.toJSON = function () { return { type: 'Buffer', data: Array.prototype.slice.call(this._arr || this, 0) } } Buffer.prototype.equals = function (b) { assert(Buffer.isBuffer(b), 'Argument must be a Buffer') return Buffer.compare(this, b) === 0 } Buffer.prototype.compare = function (b) { assert(Buffer.isBuffer(b), 'Argument must be a Buffer') return Buffer.compare(this, b) } // copy(targetBuffer, targetStart=0, sourceStart=0, sourceEnd=buffer.length) Buffer.prototype.copy = function (target, target_start, start, end) { var source = this if (!start) start = 0 if (!end && end !== 0) end = this.length if (!target_start) target_start = 0 // Copy 0 bytes; we're done if (end === start) return if (target.length === 0 || source.length === 0) return // Fatal error conditions assert(end >= start, 'sourceEnd < sourceStart') assert(target_start >= 0 && target_start < target.length, 'targetStart out of bounds') assert(start >= 0 && start < source.length, 'sourceStart out of bounds') assert(end >= 0 && end <= source.length, 'sourceEnd out of bounds') // Are we oob? if (end > this.length) end = this.length if (target.length - target_start < end - start) end = target.length - target_start + start var len = end - start if (len < 100 || !TYPED_ARRAY_SUPPORT) { for (var i = 0; i < len; i++) { target[i + target_start] = this[i + start] } } else { target._set(this.subarray(start, start + len), target_start) } } function base64Slice (buf, start, end) { if (start === 0 && end === buf.length) { return base64.fromByteArray(buf) } else { return base64.fromByteArray(buf.slice(start, end)) } } function utf8Slice (buf, start, end) { var res = '' var tmp = '' end = Math.min(buf.length, end) for (var i = start; i < end; i++) { if (buf[i] <= 0x7F) { res += decodeUtf8Char(tmp) + String.fromCharCode(buf[i]) tmp = '' } else { tmp += '%' + buf[i].toString(16) } } return res + decodeUtf8Char(tmp) } function asciiSlice (buf, start, end) { var ret = '' end = Math.min(buf.length, end) for (var i = start; i < end; i++) { ret += String.fromCharCode(buf[i]) } return ret } function binarySlice (buf, start, end) { return asciiSlice(buf, start, end) } function hexSlice (buf, start, end) { var len = buf.length if (!start || start < 0) start = 0 if (!end || end < 0 || end > len) end = len var out = '' for (var i = start; i < end; i++) { out += toHex(buf[i]) } return out } function utf16leSlice (buf, start, end) { var bytes = buf.slice(start, end) var res = '' for (var i = 0; i < bytes.length; i += 2) { res += String.fromCharCode(bytes[i] + bytes[i + 1] * 256) } return res } Buffer.prototype.slice = function (start, end) { var len = this.length start = ~~start end = end === undefined ? len : ~~end if (start < 0) { start += len; if (start < 0) start = 0 } else if (start > len) { start = len } if (end < 0) { end += len if (end < 0) end = 0 } else if (end > len) { end = len } if (end < start) end = start if (TYPED_ARRAY_SUPPORT) { return Buffer._augment(this.subarray(start, end)) } else { var sliceLen = end - start var newBuf = new Buffer(sliceLen, undefined, true) for (var i = 0; i < sliceLen; i++) { newBuf[i] = this[i + start] } return newBuf } } // `get` will be removed in Node 0.13+ Buffer.prototype.get = function (offset) { console.log('.get() is deprecated. Access using array indexes instead.') return this.readUInt8(offset) } // `set` will be removed in Node 0.13+ Buffer.prototype.set = function (v, offset) { console.log('.set() is deprecated. Access using array indexes instead.') return this.writeUInt8(v, offset) } Buffer.prototype.readUInt8 = function (offset, noAssert) { if (!noAssert) { assert(offset !== undefined && offset !== null, 'missing offset') assert(offset < this.length, 'Trying to read beyond buffer length') } if (offset >= this.length) return return this[offset] } function readUInt16 (buf, offset, littleEndian, noAssert) { if (!noAssert) { assert(typeof littleEndian === 'boolean', 'missing or invalid endian') assert(offset !== undefined && offset !== null, 'missing offset') assert(offset + 1 < buf.length, 'Trying to read beyond buffer length') } var len = buf.length if (offset >= len) return var val if (littleEndian) { val = buf[offset] if (offset + 1 < len) val |= buf[offset + 1] << 8 } else { val = buf[offset] << 8 if (offset + 1 < len) val |= buf[offset + 1] } return val } Buffer.prototype.readUInt16LE = function (offset, noAssert) { return readUInt16(this, offset, true, noAssert) } Buffer.prototype.readUInt16BE = function (offset, noAssert) { return readUInt16(this, offset, false, noAssert) } function readUInt32 (buf, offset, littleEndian, noAssert) { if (!noAssert) { assert(typeof littleEndian === 'boolean', 'missing or invalid endian') assert(offset !== undefined && offset !== null, 'missing offset') assert(offset + 3 < buf.length, 'Trying to read beyond buffer length') } var len = buf.length if (offset >= len) return var val if (littleEndian) { if (offset + 2 < len) val = buf[offset + 2] << 16 if (offset + 1 < len) val |= buf[offset + 1] << 8 val |= buf[offset] if (offset + 3 < len) val = val + (buf[offset + 3] << 24 >>> 0) } else { if (offset + 1 < len) val = buf[offset + 1] << 16 if (offset + 2 < len) val |= buf[offset + 2] << 8 if (offset + 3 < len) val |= buf[offset + 3] val = val + (buf[offset] << 24 >>> 0) } return val } Buffer.prototype.readUInt32LE = function (offset, noAssert) { return readUInt32(this, offset, true, noAssert) } Buffer.prototype.readUInt32BE = function (offset, noAssert) { return readUInt32(this, offset, false, noAssert) } Buffer.prototype.readInt8 = function (offset, noAssert) { if (!noAssert) { assert(offset !== undefined && offset !== null, 'missing offset') assert(offset < this.length, 'Trying to read beyond buffer length') } if (offset >= this.length) return var neg = this[offset] & 0x80 if (neg) return (0xff - this[offset] + 1) * -1 else return this[offset] } function readInt16 (buf, offset, littleEndian, noAssert) { if (!noAssert) { assert(typeof littleEndian === 'boolean', 'missing or invalid endian') assert(offset !== undefined && offset !== null, 'missing offset') assert(offset + 1 < buf.length, 'Trying to read beyond buffer length') } var len = buf.length if (offset >= len) return var val = readUInt16(buf, offset, littleEndian, true) var neg = val & 0x8000 if (neg) return (0xffff - val + 1) * -1 else return val } Buffer.prototype.readInt16LE = function (offset, noAssert) { return readInt16(this, offset, true, noAssert) } Buffer.prototype.readInt16BE = function (offset, noAssert) { return readInt16(this, offset, false, noAssert) } function readInt32 (buf, offset, littleEndian, noAssert) { if (!noAssert) { assert(typeof littleEndian === 'boolean', 'missing or invalid endian') assert(offset !== undefined && offset !== null, 'missing offset') assert(offset + 3 < buf.length, 'Trying to read beyond buffer length') } var len = buf.length if (offset >= len) return var val = readUInt32(buf, offset, littleEndian, true) var neg = val & 0x80000000 if (neg) return (0xffffffff - val + 1) * -1 else return val } Buffer.prototype.readInt32LE = function (offset, noAssert) { return readInt32(this, offset, true, noAssert) } Buffer.prototype.readInt32BE = function (offset, noAssert) { return readInt32(this, offset, false, noAssert) } function readFloat (buf, offset, littleEndian, noAssert) { if (!noAssert) { assert(typeof littleEndian === 'boolean', 'missing or invalid endian') assert(offset + 3 < buf.length, 'Trying to read beyond buffer length') } return ieee754.read(buf, offset, littleEndian, 23, 4) } Buffer.prototype.readFloatLE = function (offset, noAssert) { return readFloat(this, offset, true, noAssert) } Buffer.prototype.readFloatBE = function (offset, noAssert) { return readFloat(this, offset, false, noAssert) } function readDouble (buf, offset, littleEndian, noAssert) { if (!noAssert) { assert(typeof littleEndian === 'boolean', 'missing or invalid endian') assert(offset + 7 < buf.length, 'Trying to read beyond buffer length') } return ieee754.read(buf, offset, littleEndian, 52, 8) } Buffer.prototype.readDoubleLE = function (offset, noAssert) { return readDouble(this, offset, true, noAssert) } Buffer.prototype.readDoubleBE = function (offset, noAssert) { return readDouble(this, offset, false, noAssert) } Buffer.prototype.writeUInt8 = function (value, offset, noAssert) { if (!noAssert) { assert(value !== undefined && value !== null, 'missing value') assert(offset !== undefined && offset !== null, 'missing offset') assert(offset < this.length, 'trying to write beyond buffer length') verifuint(value, 0xff) } if (offset >= this.length) return this[offset] = value return offset + 1 } function writeUInt16 (buf, value, offset, littleEndian, noAssert) { if (!noAssert) { assert(value !== undefined && value !== null, 'missing value') assert(typeof littleEndian === 'boolean', 'missing or invalid endian') assert(offset !== undefined && offset !== null, 'missing offset') assert(offset + 1 < buf.length, 'trying to write beyond buffer length') verifuint(value, 0xffff) } var len = buf.length if (offset >= len) return for (var i = 0, j = Math.min(len - offset, 2); i < j; i++) { buf[offset + i] = (value & (0xff << (8 * (littleEndian ? i : 1 - i)))) >>> (littleEndian ? i : 1 - i) * 8 } return offset + 2 } Buffer.prototype.writeUInt16LE = function (value, offset, noAssert) { return writeUInt16(this, value, offset, true, noAssert) } Buffer.prototype.writeUInt16BE = function (value, offset, noAssert) { return writeUInt16(this, value, offset, false, noAssert) } function writeUInt32 (buf, value, offset, littleEndian, noAssert) { if (!noAssert) { assert(value !== undefined && value !== null, 'missing value') assert(typeof littleEndian === 'boolean', 'missing or invalid endian') assert(offset !== undefined && offset !== null, 'missing offset') assert(offset + 3 < buf.length, 'trying to write beyond buffer length') verifuint(value, 0xffffffff) } var len = buf.length if (offset >= len) return for (var i = 0, j = Math.min(len - offset, 4); i < j; i++) { buf[offset + i] = (value >>> (littleEndian ? i : 3 - i) * 8) & 0xff } return offset + 4 } Buffer.prototype.writeUInt32LE = function (value, offset, noAssert) { return writeUInt32(this, value, offset, true, noAssert) } Buffer.prototype.writeUInt32BE = function (value, offset, noAssert) { return writeUInt32(this, value, offset, false, noAssert) } Buffer.prototype.writeInt8 = function (value, offset, noAssert) { if (!noAssert) { assert(value !== undefined && value !== null, 'missing value') assert(offset !== undefined && offset !== null, 'missing offset') assert(offset < this.length, 'Trying to write beyond buffer length') verifsint(value, 0x7f, -0x80) } if (offset >= this.length) return if (value >= 0) this.writeUInt8(value, offset, noAssert) else this.writeUInt8(0xff + value + 1, offset, noAssert) return offset + 1 } function writeInt16 (buf, value, offset, littleEndian, noAssert) { if (!noAssert) { assert(value !== undefined && value !== null, 'missing value') assert(typeof littleEndian === 'boolean', 'missing or invalid endian') assert(offset !== undefined && offset !== null, 'missing offset') assert(offset + 1 < buf.length, 'Trying to write beyond buffer length') verifsint(value, 0x7fff, -0x8000) } var len = buf.length if (offset >= len) return if (value >= 0) writeUInt16(buf, value, offset, littleEndian, noAssert) else writeUInt16(buf, 0xffff + value + 1, offset, littleEndian, noAssert) return offset + 2 } Buffer.prototype.writeInt16LE = function (value, offset, noAssert) { return writeInt16(this, value, offset, true, noAssert) } Buffer.prototype.writeInt16BE = function (value, offset, noAssert) { return writeInt16(this, value, offset, false, noAssert) } function writeInt32 (buf, value, offset, littleEndian, noAssert) { if (!noAssert) { assert(value !== undefined && value !== null, 'missing value') assert(typeof littleEndian === 'boolean', 'missing or invalid endian') assert(offset !== undefined && offset !== null, 'missing offset') assert(offset + 3 < buf.length, 'Trying to write beyond buffer length') verifsint(value, 0x7fffffff, -0x80000000) } var len = buf.length if (offset >= len) return if (value >= 0) writeUInt32(buf, value, offset, littleEndian, noAssert) else writeUInt32(buf, 0xffffffff + value + 1, offset, littleEndian, noAssert) return offset + 4 } Buffer.prototype.writeInt32LE = function (value, offset, noAssert) { return writeInt32(this, value, offset, true, noAssert) } Buffer.prototype.writeInt32BE = function (value, offset, noAssert) { return writeInt32(this, value, offset, false, noAssert) } function writeFloat (buf, value, offset, littleEndian, noAssert) { if (!noAssert) { assert(value !== undefined && value !== null, 'missing value') assert(typeof littleEndian === 'boolean', 'missing or invalid endian') assert(offset !== undefined && offset !== null, 'missing offset') assert(offset + 3 < buf.length, 'Trying to write beyond buffer length') verifIEEE754(value, 3.4028234663852886e+38, -3.4028234663852886e+38) } var len = buf.length if (offset >= len) return ieee754.write(buf, value, offset, littleEndian, 23, 4) return offset + 4 } Buffer.prototype.writeFloatLE = function (value, offset, noAssert) { return writeFloat(this, value, offset, true, noAssert) } Buffer.prototype.writeFloatBE = function (value, offset, noAssert) { return writeFloat(this, value, offset, false, noAssert) } function writeDouble (buf, value, offset, littleEndian, noAssert) { if (!noAssert) { assert(value !== undefined && value !== null, 'missing value') assert(typeof littleEndian === 'boolean', 'missing or invalid endian') assert(offset !== undefined && offset !== null, 'missing offset') assert(offset + 7 < buf.length, 'Trying to write beyond buffer length') verifIEEE754(value, 1.7976931348623157E+308, -1.7976931348623157E+308) } var len = buf.length if (offset >= len) return ieee754.write(buf, value, offset, littleEndian, 52, 8) return offset + 8 } Buffer.prototype.writeDoubleLE = function (value, offset, noAssert) { return writeDouble(this, value, offset, true, noAssert) } Buffer.prototype.writeDoubleBE = function (value, offset, noAssert) { return writeDouble(this, value, offset, false, noAssert) } // fill(value, start=0, end=buffer.length) Buffer.prototype.fill = function (value, start, end) { if (!value) value = 0 if (!start) start = 0 if (!end) end = this.length assert(end >= start, 'end < start') // Fill 0 bytes; we're done if (end === start) return if (this.length === 0) return assert(start >= 0 && start < this.length, 'start out of bounds') assert(end >= 0 && end <= this.length, 'end out of bounds') var i if (typeof value === 'number') { for (i = start; i < end; i++) { this[i] = value } } else { var bytes = utf8ToBytes(value.toString()) var len = bytes.length for (i = start; i < end; i++) { this[i] = bytes[i % len] } } return this } Buffer.prototype.inspect = function () { var out = [] var len = this.length for (var i = 0; i < len; i++) { out[i] = toHex(this[i]) if (i === exports.INSPECT_MAX_BYTES) { out[i + 1] = '...' break } } return '' } /** * Creates a new `ArrayBuffer` with the *copied* memory of the buffer instance. * Added in Node 0.12. Only available in browsers that support ArrayBuffer. */ Buffer.prototype.toArrayBuffer = function () { if (typeof Uint8Array !== 'undefined') { if (TYPED_ARRAY_SUPPORT) { return (new Buffer(this)).buffer } else { var buf = new Uint8Array(this.length) for (var i = 0, len = buf.length; i < len; i += 1) { buf[i] = this[i] } return buf.buffer } } else { throw new Error('Buffer.toArrayBuffer not supported in this browser') } } // HELPER FUNCTIONS // ================ var BP = Buffer.prototype /** * Augment a Uint8Array *instance* (not the Uint8Array class!) with Buffer methods */ Buffer._augment = function (arr) { arr._isBuffer = true // save reference to original Uint8Array get/set methods before overwriting arr._get = arr.get arr._set = arr.set // deprecated, will be removed in node 0.13+ arr.get = BP.get arr.set = BP.set arr.write = BP.write arr.toString = BP.toString arr.toLocaleString = BP.toString arr.toJSON = BP.toJSON arr.equals = BP.equals arr.compare = BP.compare arr.copy = BP.copy arr.slice = BP.slice arr.readUInt8 = BP.readUInt8 arr.readUInt16LE = BP.readUInt16LE arr.readUInt16BE = BP.readUInt16BE arr.readUInt32LE = BP.readUInt32LE arr.readUInt32BE = BP.readUInt32BE arr.readInt8 = BP.readInt8 arr.readInt16LE = BP.readInt16LE arr.readInt16BE = BP.readInt16BE arr.readInt32LE = BP.readInt32LE arr.readInt32BE = BP.readInt32BE arr.readFloatLE = BP.readFloatLE arr.readFloatBE = BP.readFloatBE arr.readDoubleLE = BP.readDoubleLE arr.readDoubleBE = BP.readDoubleBE arr.writeUInt8 = BP.writeUInt8 arr.writeUInt16LE = BP.writeUInt16LE arr.writeUInt16BE = BP.writeUInt16BE arr.writeUInt32LE = BP.writeUInt32LE arr.writeUInt32BE = BP.writeUInt32BE arr.writeInt8 = BP.writeInt8 arr.writeInt16LE = BP.writeInt16LE arr.writeInt16BE = BP.writeInt16BE arr.writeInt32LE = BP.writeInt32LE arr.writeInt32BE = BP.writeInt32BE arr.writeFloatLE = BP.writeFloatLE arr.writeFloatBE = BP.writeFloatBE arr.writeDoubleLE = BP.writeDoubleLE arr.writeDoubleBE = BP.writeDoubleBE arr.fill = BP.fill arr.inspect = BP.inspect arr.toArrayBuffer = BP.toArrayBuffer return arr } var INVALID_BASE64_RE = /[^+\/0-9A-z]/g function base64clean (str) { // Node strips out invalid characters like \n and \t from the string, base64-js does not str = stringtrim(str).replace(INVALID_BASE64_RE, '') // Node allows for non-padded base64 strings (missing trailing ===), base64-js does not while (str.length % 4 !== 0) { str = str + '=' } return str } function stringtrim (str) { if (str.trim) return str.trim() return str.replace(/^\s+|\s+$/g, '') } function isArray (subject) { return (Array.isArray || function (subject) { return Object.prototype.toString.call(subject) === '[object Array]' })(subject) } function isArrayish (subject) { return isArray(subject) || Buffer.isBuffer(subject) || subject && typeof subject === 'object' && typeof subject.length === 'number' } function toHex (n) { if (n < 16) return '0' + n.toString(16) return n.toString(16) } function utf8ToBytes (str) { var byteArray = [] for (var i = 0; i < str.length; i++) { var b = str.charCodeAt(i) if (b <= 0x7F) { byteArray.push(b) } else { var start = i if (b >= 0xD800 && b <= 0xDFFF) i++ var h = encodeURIComponent(str.slice(start, i+1)).substr(1).split('%') for (var j = 0; j < h.length; j++) { byteArray.push(parseInt(h[j], 16)) } } } return byteArray } function asciiToBytes (str) { var byteArray = [] for (var i = 0; i < str.length; i++) { // Node's code seems to be doing this and not & 0x7F.. byteArray.push(str.charCodeAt(i) & 0xFF) } return byteArray } function utf16leToBytes (str) { var c, hi, lo var byteArray = [] for (var i = 0; i < str.length; i++) { c = str.charCodeAt(i) hi = c >> 8 lo = c % 256 byteArray.push(lo) byteArray.push(hi) } return byteArray } function base64ToBytes (str) { return base64.toByteArray(str) } function blitBuffer (src, dst, offset, length) { for (var i = 0; i < length; i++) { if ((i + offset >= dst.length) || (i >= src.length)) break dst[i + offset] = src[i] } return i } function decodeUtf8Char (str) { try { return decodeURIComponent(str) } catch (err) { return String.fromCharCode(0xFFFD) // UTF 8 invalid char } } /* * We have to make sure that the value is a valid integer. This means that it * is non-negative. It has no fractional component and that it does not * exceed the maximum allowed value. */ function verifuint (value, max) { assert(typeof value === 'number', 'cannot write a non-number as a number') assert(value >= 0, 'specified a negative value for writing an unsigned value') assert(value <= max, 'value is larger than maximum value for type') assert(Math.floor(value) === value, 'value has a fractional component') } function verifsint (value, max, min) { assert(typeof value === 'number', 'cannot write a non-number as a number') assert(value <= max, 'value larger than maximum allowed value') assert(value >= min, 'value smaller than minimum allowed value') assert(Math.floor(value) === value, 'value has a fractional component') } function verifIEEE754 (value, max, min) { assert(typeof value === 'number', 'cannot write a non-number as a number') assert(value <= max, 'value larger than maximum allowed value') assert(value >= min, 'value smaller than minimum allowed value') } function assert (test, message) { if (!test) throw new Error(message || 'Failed assertion') } },{"base64-js":89,"ieee754":90}],89:[function(require,module,exports){ var lookup = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/'; ;(function (exports) { 'use strict'; var Arr = (typeof Uint8Array !== 'undefined') ? Uint8Array : Array var PLUS = '+'.charCodeAt(0) var SLASH = '/'.charCodeAt(0) var NUMBER = '0'.charCodeAt(0) var LOWER = 'a'.charCodeAt(0) var UPPER = 'A'.charCodeAt(0) function decode (elt) { var code = elt.charCodeAt(0) if (code === PLUS) return 62 // '+' if (code === SLASH) return 63 // '/' if (code < NUMBER) return -1 //no match if (code < NUMBER + 10) return code - NUMBER + 26 + 26 if (code < UPPER + 26) return code - UPPER if (code < LOWER + 26) return code - LOWER + 26 } function b64ToByteArray (b64) { var i, j, l, tmp, placeHolders, arr if (b64.length % 4 > 0) { throw new Error('Invalid string. Length must be a multiple of 4') } // the number of equal signs (place holders) // if there are two placeholders, than the two characters before it // represent one byte // if there is only one, then the three characters before it represent 2 bytes // this is just a cheap hack to not do indexOf twice var len = b64.length placeHolders = '=' === b64.charAt(len - 2) ? 2 : '=' === b64.charAt(len - 1) ? 1 : 0 // base64 is 4/3 + up to two characters of the original data arr = new Arr(b64.length * 3 / 4 - placeHolders) // if there are placeholders, only get up to the last complete 4 chars l = placeHolders > 0 ? b64.length - 4 : b64.length var L = 0 function push (v) { arr[L++] = v } for (i = 0, j = 0; i < l; i += 4, j += 3) { tmp = (decode(b64.charAt(i)) << 18) | (decode(b64.charAt(i + 1)) << 12) | (decode(b64.charAt(i + 2)) << 6) | decode(b64.charAt(i + 3)) push((tmp & 0xFF0000) >> 16) push((tmp & 0xFF00) >> 8) push(tmp & 0xFF) } if (placeHolders === 2) { tmp = (decode(b64.charAt(i)) << 2) | (decode(b64.charAt(i + 1)) >> 4) push(tmp & 0xFF) } else if (placeHolders === 1) { tmp = (decode(b64.charAt(i)) << 10) | (decode(b64.charAt(i + 1)) << 4) | (decode(b64.charAt(i + 2)) >> 2) push((tmp >> 8) & 0xFF) push(tmp & 0xFF) } return arr } function uint8ToBase64 (uint8) { var i, extraBytes = uint8.length % 3, // if we have 1 byte left, pad 2 bytes output = "", temp, length function encode (num) { return lookup.charAt(num) } function tripletToBase64 (num) { return encode(num >> 18 & 0x3F) + encode(num >> 12 & 0x3F) + encode(num >> 6 & 0x3F) + encode(num & 0x3F) } // go through the array every three bytes, we'll deal with trailing stuff later for (i = 0, length = uint8.length - extraBytes; i < length; i += 3) { temp = (uint8[i] << 16) + (uint8[i + 1] << 8) + (uint8[i + 2]) output += tripletToBase64(temp) } // pad the end with zeros, but make sure to not forget the extra bytes switch (extraBytes) { case 1: temp = uint8[uint8.length - 1] output += encode(temp >> 2) output += encode((temp << 4) & 0x3F) output += '==' break case 2: temp = (uint8[uint8.length - 2] << 8) + (uint8[uint8.length - 1]) output += encode(temp >> 10) output += encode((temp >> 4) & 0x3F) output += encode((temp << 2) & 0x3F) output += '=' break } return output } exports.toByteArray = b64ToByteArray exports.fromByteArray = uint8ToBase64 }(typeof exports === 'undefined' ? (this.base64js = {}) : exports)) },{}],90:[function(require,module,exports){ exports.read = function(buffer, offset, isLE, mLen, nBytes) { var e, m, eLen = nBytes * 8 - mLen - 1, eMax = (1 << eLen) - 1, eBias = eMax >> 1, nBits = -7, i = isLE ? (nBytes - 1) : 0, d = isLE ? -1 : 1, s = buffer[offset + i]; i += d; e = s & ((1 << (-nBits)) - 1); s >>= (-nBits); nBits += eLen; for (; nBits > 0; e = e * 256 + buffer[offset + i], i += d, nBits -= 8); m = e & ((1 << (-nBits)) - 1); e >>= (-nBits); nBits += mLen; for (; nBits > 0; m = m * 256 + buffer[offset + i], i += d, nBits -= 8); if (e === 0) { e = 1 - eBias; } else if (e === eMax) { return m ? NaN : ((s ? -1 : 1) * Infinity); } else { m = m + Math.pow(2, mLen); e = e - eBias; } return (s ? -1 : 1) * m * Math.pow(2, e - mLen); }; exports.write = function(buffer, value, offset, isLE, mLen, nBytes) { var e, m, c, eLen = nBytes * 8 - mLen - 1, eMax = (1 << eLen) - 1, eBias = eMax >> 1, rt = (mLen === 23 ? Math.pow(2, -24) - Math.pow(2, -77) : 0), i = isLE ? 0 : (nBytes - 1), d = isLE ? 1 : -1, s = value < 0 || (value === 0 && 1 / value < 0) ? 1 : 0; value = Math.abs(value); if (isNaN(value) || value === Infinity) { m = isNaN(value) ? 1 : 0; e = eMax; } else { e = Math.floor(Math.log(value) / Math.LN2); if (value * (c = Math.pow(2, -e)) < 1) { e--; c *= 2; } if (e + eBias >= 1) { value += rt / c; } else { value += rt * Math.pow(2, 1 - eBias); } if (value * c >= 2) { e++; c /= 2; } if (e + eBias >= eMax) { m = 0; e = eMax; } else if (e + eBias >= 1) { m = (value * c - 1) * Math.pow(2, mLen); e = e + eBias; } else { m = value * Math.pow(2, eBias - 1) * Math.pow(2, mLen); e = 0; } } for (; mLen >= 8; buffer[offset + i] = m & 0xff, i += d, m /= 256, mLen -= 8); e = (e << mLen) | m; eLen += mLen; for (; eLen > 0; buffer[offset + i] = e & 0xff, i += d, e /= 256, eLen -= 8); buffer[offset + i - d] |= s * 128; }; },{}],91:[function(require,module,exports){ // (c) Dean McNamee , 2012. // // https://github.com/deanm/css-color-parser-js // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to // deal in the Software without restriction, including without limitation the // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or // sell copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS // IN THE SOFTWARE. // http://www.w3.org/TR/css3-color/ var kCSSColorTable = { "transparent": [0,0,0,0], "aliceblue": [240,248,255,1], "antiquewhite": [250,235,215,1], "aqua": [0,255,255,1], "aquamarine": [127,255,212,1], "azure": [240,255,255,1], "beige": [245,245,220,1], "bisque": [255,228,196,1], "black": [0,0,0,1], "blanchedalmond": [255,235,205,1], "blue": [0,0,255,1], "blueviolet": [138,43,226,1], "brown": [165,42,42,1], "burlywood": [222,184,135,1], "cadetblue": [95,158,160,1], "chartreuse": [127,255,0,1], "chocolate": [210,105,30,1], "coral": [255,127,80,1], "cornflowerblue": [100,149,237,1], "cornsilk": [255,248,220,1], "crimson": [220,20,60,1], "cyan": [0,255,255,1], "darkblue": [0,0,139,1], "darkcyan": [0,139,139,1], "darkgoldenrod": [184,134,11,1], "darkgray": [169,169,169,1], "darkgreen": [0,100,0,1], "darkgrey": [169,169,169,1], "darkkhaki": [189,183,107,1], "darkmagenta": [139,0,139,1], "darkolivegreen": [85,107,47,1], "darkorange": [255,140,0,1], "darkorchid": [153,50,204,1], "darkred": [139,0,0,1], "darksalmon": [233,150,122,1], "darkseagreen": [143,188,143,1], "darkslateblue": [72,61,139,1], "darkslategray": [47,79,79,1], "darkslategrey": [47,79,79,1], "darkturquoise": [0,206,209,1], "darkviolet": [148,0,211,1], "deeppink": [255,20,147,1], "deepskyblue": [0,191,255,1], "dimgray": [105,105,105,1], "dimgrey": [105,105,105,1], "dodgerblue": [30,144,255,1], "firebrick": [178,34,34,1], "floralwhite": [255,250,240,1], "forestgreen": [34,139,34,1], "fuchsia": [255,0,255,1], "gainsboro": [220,220,220,1], "ghostwhite": [248,248,255,1], "gold": [255,215,0,1], "goldenrod": [218,165,32,1], "gray": [128,128,128,1], "green": [0,128,0,1], "greenyellow": [173,255,47,1], "grey": [128,128,128,1], "honeydew": [240,255,240,1], "hotpink": [255,105,180,1], "indianred": [205,92,92,1], "indigo": [75,0,130,1], "ivory": [255,255,240,1], "khaki": [240,230,140,1], "lavender": [230,230,250,1], "lavenderblush": [255,240,245,1], "lawngreen": [124,252,0,1], "lemonchiffon": [255,250,205,1], "lightblue": [173,216,230,1], "lightcoral": [240,128,128,1], "lightcyan": [224,255,255,1], "lightgoldenrodyellow": [250,250,210,1], "lightgray": [211,211,211,1], "lightgreen": [144,238,144,1], "lightgrey": [211,211,211,1], "lightpink": [255,182,193,1], "lightsalmon": [255,160,122,1], "lightseagreen": [32,178,170,1], "lightskyblue": [135,206,250,1], "lightslategray": [119,136,153,1], "lightslategrey": [119,136,153,1], "lightsteelblue": [176,196,222,1], "lightyellow": [255,255,224,1], "lime": [0,255,0,1], "limegreen": [50,205,50,1], "linen": [250,240,230,1], "magenta": [255,0,255,1], "maroon": [128,0,0,1], "mediumaquamarine": [102,205,170,1], "mediumblue": [0,0,205,1], "mediumorchid": [186,85,211,1], "mediumpurple": [147,112,219,1], "mediumseagreen": [60,179,113,1], "mediumslateblue": [123,104,238,1], "mediumspringgreen": [0,250,154,1], "mediumturquoise": [72,209,204,1], "mediumvioletred": [199,21,133,1], "midnightblue": [25,25,112,1], "mintcream": [245,255,250,1], "mistyrose": [255,228,225,1], "moccasin": [255,228,181,1], "navajowhite": [255,222,173,1], "navy": [0,0,128,1], "oldlace": [253,245,230,1], "olive": [128,128,0,1], "olivedrab": [107,142,35,1], "orange": [255,165,0,1], "orangered": [255,69,0,1], "orchid": [218,112,214,1], "palegoldenrod": [238,232,170,1], "palegreen": [152,251,152,1], "paleturquoise": [175,238,238,1], "palevioletred": [219,112,147,1], "papayawhip": [255,239,213,1], "peachpuff": [255,218,185,1], "peru": [205,133,63,1], "pink": [255,192,203,1], "plum": [221,160,221,1], "powderblue": [176,224,230,1], "purple": [128,0,128,1], "red": [255,0,0,1], "rosybrown": [188,143,143,1], "royalblue": [65,105,225,1], "saddlebrown": [139,69,19,1], "salmon": [250,128,114,1], "sandybrown": [244,164,96,1], "seagreen": [46,139,87,1], "seashell": [255,245,238,1], "sienna": [160,82,45,1], "silver": [192,192,192,1], "skyblue": [135,206,235,1], "slateblue": [106,90,205,1], "slategray": [112,128,144,1], "slategrey": [112,128,144,1], "snow": [255,250,250,1], "springgreen": [0,255,127,1], "steelblue": [70,130,180,1], "tan": [210,180,140,1], "teal": [0,128,128,1], "thistle": [216,191,216,1], "tomato": [255,99,71,1], "turquoise": [64,224,208,1], "violet": [238,130,238,1], "wheat": [245,222,179,1], "white": [255,255,255,1], "whitesmoke": [245,245,245,1], "yellow": [255,255,0,1], "yellowgreen": [154,205,50,1]} function clamp_css_byte(i) { // Clamp to integer 0 .. 255. i = Math.round(i); // Seems to be what Chrome does (vs truncation). return i < 0 ? 0 : i > 255 ? 255 : i; } function clamp_css_float(f) { // Clamp to float 0.0 .. 1.0. return f < 0 ? 0 : f > 1 ? 1 : f; } function parse_css_int(str) { // int or percentage. if (str[str.length - 1] === '%') return clamp_css_byte(parseFloat(str) / 100 * 255); return clamp_css_byte(parseInt(str)); } function parse_css_float(str) { // float or percentage. if (str[str.length - 1] === '%') return clamp_css_float(parseFloat(str) / 100); return clamp_css_float(parseFloat(str)); } function css_hue_to_rgb(m1, m2, h) { if (h < 0) h += 1; else if (h > 1) h -= 1; if (h * 6 < 1) return m1 + (m2 - m1) * h * 6; if (h * 2 < 1) return m2; if (h * 3 < 2) return m1 + (m2 - m1) * (2/3 - h) * 6; return m1; } function parseCSSColor(css_str) { // Remove all whitespace, not compliant, but should just be more accepting. var str = css_str.replace(/ /g, '').toLowerCase(); // Color keywords (and transparent) lookup. if (str in kCSSColorTable) return kCSSColorTable[str].slice(); // dup. // #abc and #abc123 syntax. if (str[0] === '#') { if (str.length === 4) { var iv = parseInt(str.substr(1), 16); // TODO(deanm): Stricter parsing. if (!(iv >= 0 && iv <= 0xfff)) return null; // Covers NaN. return [((iv & 0xf00) >> 4) | ((iv & 0xf00) >> 8), (iv & 0xf0) | ((iv & 0xf0) >> 4), (iv & 0xf) | ((iv & 0xf) << 4), 1]; } else if (str.length === 7) { var iv = parseInt(str.substr(1), 16); // TODO(deanm): Stricter parsing. if (!(iv >= 0 && iv <= 0xffffff)) return null; // Covers NaN. return [(iv & 0xff0000) >> 16, (iv & 0xff00) >> 8, iv & 0xff, 1]; } return null; } var op = str.indexOf('('), ep = str.indexOf(')'); if (op !== -1 && ep + 1 === str.length) { var fname = str.substr(0, op); var params = str.substr(op+1, ep-(op+1)).split(','); var alpha = 1; // To allow case fallthrough. switch (fname) { case 'rgba': if (params.length !== 4) return null; alpha = parse_css_float(params.pop()); // Fall through. case 'rgb': if (params.length !== 3) return null; return [parse_css_int(params[0]), parse_css_int(params[1]), parse_css_int(params[2]), alpha]; case 'hsla': if (params.length !== 4) return null; alpha = parse_css_float(params.pop()); // Fall through. case 'hsl': if (params.length !== 3) return null; var h = (((parseFloat(params[0]) % 360) + 360) % 360) / 360; // 0 .. 1 // NOTE(deanm): According to the CSS spec s/l should only be // percentages, but we don't bother and let float or percentage. var s = parse_css_float(params[1]); var l = parse_css_float(params[2]); var m2 = l <= 0.5 ? l * (s + 1) : l + s - l * s; var m1 = l * 2 - m2; return [clamp_css_byte(css_hue_to_rgb(m1, m2, h+1/3) * 255), clamp_css_byte(css_hue_to_rgb(m1, m2, h) * 255), clamp_css_byte(css_hue_to_rgb(m1, m2, h-1/3) * 255), alpha]; default: return null; } } return null; } try { exports.parseCSSColor = parseCSSColor } catch(e) { } },{}],92:[function(require,module,exports){ var geojsonArea = require('geojson-area'); module.exports = rewind; function rewind(gj, outer) { switch ((gj && gj.type) || null) { case 'FeatureCollection': gj.features = gj.features.map(curryOuter(rewind, outer)); return gj; case 'Feature': gj.geometry = rewind(gj.geometry, outer); return gj; case 'Polygon': case 'MultiPolygon': return correct(gj, outer); default: return gj; } } function curryOuter(a, b) { return function(_) { return a(_, b); }; } function correct(_, outer) { if (_.type === 'Polygon') { _.coordinates = correctRings(_.coordinates, outer); } else if (_.type === 'MultiPolygon') { _.coordinates = _.coordinates.map(curryOuter(correctRings, outer)); } return _; } function correctRings(_, outer) { outer = !!outer; _[0] = wind(_[0], !outer); for (var i = 1; i < _.length; i++) { _[i] = wind(_[i], outer); } return _; } function wind(_, dir) { return cw(_) === dir ? _ : _.reverse(); } function cw(_) { return geojsonArea.ring(_) >= 0; } },{"geojson-area":93}],93:[function(require,module,exports){ var wgs84 = require('wgs84'); module.exports.geometry = geometry; module.exports.ring = ringArea; function geometry(_) { if (_.type === 'Polygon') return polygonArea(_.coordinates); else if (_.type === 'MultiPolygon') { var area = 0; for (var i = 0; i < _.coordinates.length; i++) { area += polygonArea(_.coordinates[i]); } return area; } else { return null; } } function polygonArea(coords) { var area = 0; if (coords && coords.length > 0) { area += Math.abs(ringArea(coords[0])); for (var i = 1; i < coords.length; i++) { area -= Math.abs(ringArea(coords[i])); } } return area; } /** * Calculate the approximate area of the polygon were it projected onto * the earth. Note that this area will be positive if ring is oriented * clockwise, otherwise it will be negative. * * Reference: * Robert. G. Chamberlain and William H. Duquette, "Some Algorithms for * Polygons on a Sphere", JPL Publication 07-03, Jet Propulsion * Laboratory, Pasadena, CA, June 2007 http://trs-new.jpl.nasa.gov/dspace/handle/2014/40409 * * Returns: * {float} The approximate signed geodesic area of the polygon in square * meters. */ function ringArea(coords) { var area = 0; if (coords.length > 2) { var p1, p2; for (var i = 0; i < coords.length - 1; i++) { p1 = coords[i]; p2 = coords[i + 1]; area += rad(p2[0] - p1[0]) * (2 + Math.sin(rad(p1[1])) + Math.sin(rad(p2[1]))); } area = area * wgs84.RADIUS * wgs84.RADIUS / 2; } return area; } function rad(_) { return _ * Math.PI / 180; } },{"wgs84":94}],94:[function(require,module,exports){ module.exports.RADIUS = 6378137; module.exports.FLATTENING = 1/298.257223563; module.exports.POLAR_RADIUS = 6356752.3142; },{}],95:[function(require,module,exports){ module.exports={ "$version": 4, "$root": { "version": { "required": true, "type": "enum", "values": [ 4 ], "doc": "Stylesheet version number. Must be 4." }, "constants": { "type": "constants", "doc": "An object of constants to be referenced in layers." }, "sources": { "required": true, "type": "sources", "doc": "Data source specifications for layers to pull from." }, "layers": { "required": true, "type": "array", "value": "layer", "doc": "An array of layers. The order of layers coincides with the order they will be drawn." }, "sprite": { "type": "sprite", "doc": "Sprite definition." }, "glyphs": { "type": "string", "doc": "A URL template for loading signed-distance-field glyph sets in PBF format. Valid tokens are {fontstack} and {range}." }, "transition": { "type": "transition", "doc": "A global transition definition to use as a default across properties." } }, "sprite": [{ "type": "string", "doc": "A base URL for retrieving the sprite image and metadata. The extensions `.png`, `.json` and scale factor `@2x.png` will be automatically appended." }], "constants": { "*": { "type": "*", "doc": "A constant that will be replaced verbatim in the referencing place. This can be anything, including objects and arrays. All variable names must be prefixed with an `@` symbol." } }, "sources": { "*": { "type": "source" } }, "source": { "type": { "required": true, "type": "enum", "values": [ "vector", "raster", "geojson", "video" ], "doc": "The data type of the source." }, "url": { "required": true, "type": "string", "doc": "A URL, or URL template to retrive the source data." }, "tileSize": { "type": "number", "default": 512, "doc": "The minimum visual size (in px) to display tiles for this layer. Only configurable for raster layers." }, "minZoom": { "type": "number", "default": 0, "doc": "Minimum zoom level for which tiles are available." }, "maxZoom": { "type": "number", "default": 22, "doc": "Maximum zoom level for which tiles are available. Data from tiles at the maxZoom are used when displaying the map at higher zoom levels." }, "*": { "type": "*", "doc": "Other keys to configure the data source." } }, "layer": { "id": { "type": "string", "doc": "Unique layer name." }, "type": { "type": "enum", "values": [ "fill", "line", "symbol", "raster", "background" ], "doc": "Rendering type of this layer." }, "ref": { "type": "string", "doc": "References another layer to copy `source`, `source_layer`, `filter`, and `render` properties from. This allows the layers to share processing and be more efficient." }, "source": { "type": "string", "doc": "Name of a source description to be used for this layer." }, "source-layer": { "type": "string", "doc": "Layer to use from a vector tile source. Required if the source supports multiple layers." }, "min-zoom": { "type": "number", "doc": "The minimum zoom level on which the layer gets parsed and appears on." }, "max-zoom": { "type": "number", "doc": "The maximum zoom level on which the layer gets parsed and appears on." }, "interactive": { "type": "boolean", "doc": "Enable querying of feature data from this layer for interactivity.", "default": false }, "render": { "type": "render", "doc": "Symbolizer type that should be used to visualize this layer. If unspecified or null, this layer is not treated as a symbolizer and only exists to have properties inherited to other layers using ref." }, "filter": { "type": "filter", "doc": "Array or object of filters or expressions." }, "layers": { "type": "array", "value": "layer", "doc": "If `type` is `raster`, the child layers are composited together onto the previous level layer level." }, "style": { "type": "class", "doc": "Default style properties for this layer." }, "style.*": { "type": "class", "doc": "Override style properties for this layer. The class name is the part after the first dot." } }, "render": [ "render_fill", "render_line", "render_symbol", "render_raster", "render_background" ], "render_background": { }, "render_fill": { }, "render_line": { "line-cap": { "type": "enum", "values": [ "butt", "round", "square" ], "default": "butt", "doc": "The display of line endings." }, "line-join": { "type": "enum", "values": [ "bevel", "round", "miter" ], "default": "miter", "doc": "The display of lines when joining." }, "line-miter-limit": { "type": "number", "default": 2, "doc": "Used to automatically convert miter joins to bevel joins for sharp angles." }, "line-round-limit": { "type": "number", "default": 1, "doc": "Used to automatically convert round joins to miter joins for shallow angles." } }, "render_symbol": { "symbol-placement": { "type": "enum", "values": [ "point", "line" ], "default": "point", "doc": "Placement of a label relative to its geometry. `Line` can only be used on LineStrings and Polygons." }, "symbol-min-distance": { "type": "number", "default": 250, "doc": "Minimum distance between two symbol anchors (px)" }, "symbol-avoid-edges": { "type": "boolean", "default": false, "doc": "If true, the symbols will not cross tile edges. Symbols that cross tile edges may cause collisions in some cases. This property should be set to true if the layer does not have enough padding in the vector tile to prevent collisions, or if it is a point symbol layer placed after a line symbol layer." }, "icon-allow-overlap": { "type": "boolean", "default": false, "doc": "If true, the icon will be visible even if it collides with other icons and text." }, "icon-ignore-placement": { "type": "boolean", "default": false, "doc": "If true, the icon won't affect placement of other icons and text." }, "icon-optional": { "type": "boolean", "default": false, "doc": "If true, text can be shown without its corresponding icon." }, "icon-rotation-alignment": { "type": "enum", "values": [ "map", "viewport" ], "default": "viewport", "doc": "Orientation of icon when map is rotated" }, "icon-max-size": { "type": "number", "default": 1, "doc": "The maximum amount to scale the icon by." }, "icon-image": { "type": "string", "doc": "A string with {tokens} replaced, referencing the data property to pull from." }, "icon-rotate": { "type": "number", "default": 0, "doc": "Rotates the icon clockwise by the specified number of degrees." }, "icon-padding": { "type": "number", "default": 2, "doc": "Padding value around icon bounding box to avoid icon collisions (px)." }, "icon-keep-upright": { "type": "boolean", "default": false, "doc": "If true, the icon may be flipped to prevent it from being rendered upside-down" }, "icon-offset": { "type": "array", "value": "number", "length": 2, "default": [ 0, 0 ], "doc": "Icon's offset, in pixels. Values are [x, y] where negatives indicate left and up, respectively." }, "text-rotation-alignment": { "type": "enum", "values": [ "map", "viewport" ], "default": "viewport", "doc": "Orientation of icon or text when map is rotated" }, "text-field": { "type": "string", "default": "", "doc": "Value to use for a text label. Feature properties are specified using tokens like {field_name}." }, "text-font": { "type": "string", "doc": "Font stack to use for displaying text." }, "text-max-size": { "type": "number", "default": 16, "doc": "The maximum size text will be displayed." }, "text-max-width": { "type": "number", "default": 15, "doc": "The maximum line width for text wrapping (em)." }, "text-line-height": { "type": "number", "default": 1.2, "doc": "Text leading value for multi-line text." }, "text-letter-spacing": { "type": "number", "default": 0, "doc": "Text kerning value (em)." }, "text-justify": { "type": "enum", "values": [ "center", "left", "right" ], "default": "center", "doc": "Text justification options." }, "text-horizontal-align": { "type": "enum", "values": [ "left", "center", "right" ], "default": "center", "doc": "Horizontal alignment of the text relative to the anchor." }, "text-vertical-align": { "type": "enum", "values": [ "top", "center", "bottom" ], "default": "center", "doc": "Vertical alignment of the text relative to the anchor." }, "text-max-angle": { "type": "number", "default": 45, "doc": "The maximum angle change, in degrees, allowed between adjacent characters." }, "text-rotate": { "type": "number", "default": 0, "doc": "Rotates the text clockwise by the specified number of degrees." }, "text-padding": { "type": "number", "default": 2, "doc": "Padding value around text bounding box to avoid label collisions (px)." }, "text-keep-upright": { "type": "boolean", "default": true, "doc": "If true, the direction of the text may be flipped to prevent it from being rendered upside-down" }, "text-transform": { "type": "enum", "values": [ "none", "uppercase", "lowercase" ], "default": "none", "doc": "Specifies how to capitalize text, similar to the CSS `text-transform` property." }, "text-offset": { "type": "array", "value": "number", "length": 2, "default": [ 0, 0 ] }, "text-allow-overlap": { "type": "boolean", "default": false, "doc": "If true, the text will be visible even if it collides with other icons and labels." }, "text-ignore-placement": { "type": "boolean", "default": false, "doc": "If true, the text won't affect placement of other icons and labels." }, "text-optional": { "type": "boolean", "default": false, "doc": "If true, icons can be shown without their corresponding text." } }, "render_raster": { "raster-size": { "type": "number", "function": true, "default": 256, "doc": "The texture image size (in pixels) vector layers will be rasterized at. Will automatically by scaled to match the visual tile size." }, "raster-blur": { "type": "number", "function": true, "default": 0, "doc": "Blur radius to apply to the raster texture before display." } }, "filter": [ { "type": "filter_expression", "doc": "Various filter expressions. Unless overridden by parent, these are interpreted as `AND`." }, { "type": "array", "value": "filter_expression", "doc": "Various filter expressions. Unless overridden by parent, these are interpreted as `OR`." } ], "filter_expression": { "&": { "type": "filter", "doc": "AND operator." }, "|": { "type": "filter", "doc": "OR operator." }, "^": { "type": "filter", "doc": "XOR operator." }, "!": { "type": "filter", "doc": "NOR operator." }, "$type": { "type": "enum", "values": [ "Point", "LineString", "Polygon" ], "doc": "Geometry type that features must match." }, "*": [ { "type": "filter_comparison", "doc": "Arbitarily named feature member. A comparison object defining a filter expression." }, { "type": "filter_value", "doc": "Arbitarily named feature member. A filter_value implies the equality (string/number/boolean) or set membership operator (array)." } ] }, "filter_comparison": { "==": { "type": "filter_value", "doc": "Equality operator." }, "!=": { "type": "filter_value", "doc": "Inequality operator." }, ">": { "type": "filter_value", "doc": "Greater than operator." }, ">=": { "type": "filter_value", "doc": "Greater or equal than operator." }, "<": { "type": "filter_value", "doc": "Less than operator." }, "<=": { "type": "filter_value", "doc": "Less than or equal operator." }, "in": { "type": "array", "value": "filter_primitive", "doc": "Set member operator." }, "!in": { "type": "array", "value": "filter_primitive", "doc": "Not in set operator." } }, "filter_value": [ { "type": "filter_primitive" }, { "type": "array", "value": "filter_primitive" } ], "filter_primitive": [ { "type": "string" }, { "type": "number" }, { "type": "boolean" } ], "function": { "stops": { "type": "array", "required": true, "doc": "An array of stops.", "value": "function_stop" }, "base": { "type": "number", "default": 1, "doc": "The exponential base of the interpolation curve. It controls the rate at which the result increases. Higher values make the result increase more towards the high end of the range. With `1` the stops are interpolated linearly." } }, "function_stop": { "type": "array", "value": ["number", "color"], "length": 2, "doc": "Zoom level and value pair." }, "class": [ "class_fill", "class_line", "class_symbol", "class_raster", "class_background" ], "class_fill": { "fill-antialias": { "type": "boolean", "default": true, "function": true, "doc": "Whether or not the fill should be antialiased." }, "fill-opacity": { "type": "number", "function": true, "default": 1, "transition": true }, "fill-color": { "type": "color", "default": [ 0, 0, 0, 1 ], "function": true, "transition": true }, "fill-outline-color": { "type": "color", "doc": "The outline color of the fill. Matches the value of `fill-color` if unspecified.", "function": true, "transition": true }, "fill-translate": { "type": "array", "value": "number", "length": 2, "default": [ 0, 0 ], "function": true, "transition": true, "doc": "The geometry's offset, in pixels. Values are [x, y] where negatives indicate left and up, respectively." }, "fill-translate-anchor": { "type": "enum", "values": [ "map", "viewport" ], "doc": "Control whether the translation is relative to the map (north) or viewport (screen)", "default": "map" }, "fill-image": { "type": "string", "doc": "Name of image in sprite to use for drawing image fills." } }, "class_line": { "line-opacity": { "type": "number", "function": true, "default": 1, "transition": true }, "line-color": { "type": "color", "default": [ 0, 0, 0, 1 ], "function": true, "transition": true }, "line-translate": { "type": "array", "value": "number", "length": 2, "default": [ 0, 0 ], "function": true, "transition": true, "doc": "The geometry's offset, in pixels. Values are [x, y] where negatives indicate left and up, respectively." }, "line-translate-anchor": { "type": "enum", "values": [ "map", "viewport" ], "doc": "Control whether the translation is relative to the map (north) or viewport (screen)", "default": "map" }, "line-width": { "type": "number", "default": 1, "function": true, "transition": true, "doc": "Line width (in px)" }, "line-offset": { "type": "number", "default": 0, "doc": "Line casing where `line-offset` indicates total width. @TODO rename?", "function": true, "transition": true }, "line-blur": { "type": "number", "default": 0, "function": true, "transition": true, "doc": "Line blur, in pixels." }, "line-dasharray": { "type": "array", "value": "number", "length": 2, "default": [ 1, -1 ], "function": true, "transition": true }, "line-image": { "type": "string", "doc": "Name of image in sprite to use for drawing image lines." } }, "class_symbol": { "icon-opacity": { "type": "number", "default": 1, "function": true, "transition": true }, "icon-size": { "type": "number", "default": 1, "function": true, "transition": true, "doc": "The amount to scale the icon by. 1 is original size, 3 triples the size." }, "icon-color": { "type": "color", "default": [ 0, 0, 0, 1 ], "function": true, "transition": true, "doc": "The color of the icon. This can only be used with sdf icons." }, "icon-halo-color": { "type": "color", "default": [ 0, 0, 0, 0 ], "function": true, "transition": true, "doc": "The color of the icon's halo. Icon halos can only be used with sdf icons." }, "icon-halo-width": { "type": "number", "default": 0, "function": true, "transition": true, "doc": "How far away the halo is from the icon outline, in pixels." }, "icon-halo-blur": { "type": "number", "default": 0, "function": true, "transition": true, "doc": "Fade out the halo towards the outside, in pixels." }, "icon-translate": { "type": "array", "value": "number", "length": 2, "default": [ 0, 0 ], "function": true, "transition": true, "doc": "An icon's offset, in pixels. Values are [x, y] where negatives indicate left and up, respectively." }, "icon-translate-anchor": { "type": "enum", "values": [ "map", "viewport" ], "doc": "Control whether the translation is relative to the map (north) or viewport (screen)", "default": "map" }, "text-opacity": { "type": "number", "default": 1, "function": true, "transition": true }, "text-size": { "type": "number", "default": 16, "function": true, "transition": true, "doc": "Font size in pixels. If unspecified, the text will be as big as allowed by the layer definition." }, "text-color": { "type": "color", "default": [ 0, 0, 0, 1 ], "function": true, "transition": true }, "text-halo-color": { "type": "color", "default": [ 0, 0, 0, 0 ], "function": true, "transition": true }, "text-halo-width": { "type": "number", "default": 0, "function": true, "transition": true, "doc": "How far away the halo is from the font outline, in pixels. Max text halo width is 1/4 of the font-size (px)." }, "text-halo-blur": { "type": "number", "default": 0, "function": true, "transition": true, "doc": "Fade out the halo towards the outside, in pixels." }, "text-translate": { "type": "array", "value": "number", "length": 2, "default": [ 0, 0 ], "function": true, "transition": true, "doc": "A label's offset, in pixels. Values are [x, y] where negatives indicate left and up, respectively." }, "text-translate-anchor": { "type": "enum", "values": [ "map", "viewport" ], "doc": "Control whether the translation is relative to the map (north) or viewport (screen)", "default": "map" } }, "class_raster": { "raster-opacity": { "type": "number", "default": 1, "transition": true }, "raster-hue-rotate": { "type": "number", "default": 0, "function": true, "transition": true, "doc": "Rotates hues around the color wheel by the specified number of degrees." }, "raster-brightness": { "type": "array", "value": "number", "length": 2, "default": [ 0, 1 ], "function": true, "transition": true }, "raster-saturation": { "type": "number", "default": 0, "function": true, "transition": true }, "raster-contrast": { "type": "number", "default": 0, "function": true, "transition": true }, "raster-fade": { "type": "number", "default": 0, "function": true, "transition": true, "doc": "Duration of the fade when a new tile is added. @TODO rename?" } }, "class_background": { "background-color": { "type": "color", "default": [ 0, 0, 0, 1 ], "function": true, "transition": true }, "background-image": { "type": "string" } }, "transition": { "duration": { "type": "number", "default": 300, "doc": "Time in milliseconds that it takes for transitions to complete." }, "delay": { "type": "number", "default": 0, "doc": "Time in milliseconds before a transition begins." } } } },{}],96:[function(require,module,exports){ (function (Buffer){ 'use strict'; var ieee754 = require('ieee754'); module.exports = Protobuf; function Protobuf(buf) { this.buf = buf; this.pos = 0; } Protobuf.prototype = { get length() { return this.buf.length; } }; Protobuf.Varint = 0; Protobuf.Int64 = 1; Protobuf.Message = 2; Protobuf.String = 2; Protobuf.Packed = 2; Protobuf.Int32 = 5; Protobuf.prototype.destroy = function() { this.buf = null; }; // === READING ================================================================= Protobuf.prototype.readUInt32 = function() { var val = this.buf.readUInt32LE(this.pos); this.pos += 4; return val; }; Protobuf.prototype.readUInt64 = function() { var val = this.buf.readUInt64LE(this.pos); this.pos += 8; return val; }; Protobuf.prototype.readDouble = function() { var val = ieee754.read(this.buf, this.pos, true, 52, 8); this.pos += 8; return val; }; Protobuf.prototype.readVarint = function() { // TODO: bounds checking var pos = this.pos; if (this.buf[pos] <= 0x7f) { this.pos++; return this.buf[pos]; } else if (this.buf[pos + 1] <= 0x7f) { this.pos += 2; return (this.buf[pos] & 0x7f) | (this.buf[pos + 1] << 7); } else if (this.buf[pos + 2] <= 0x7f) { this.pos += 3; return (this.buf[pos] & 0x7f) | (this.buf[pos + 1] & 0x7f) << 7 | (this.buf[pos + 2]) << 14; } else if (this.buf[pos + 3] <= 0x7f) { this.pos += 4; return (this.buf[pos] & 0x7f) | (this.buf[pos + 1] & 0x7f) << 7 | (this.buf[pos + 2] & 0x7f) << 14 | (this.buf[pos + 3]) << 21; } else if (this.buf[pos + 4] <= 0x7f) { this.pos += 5; return ((this.buf[pos] & 0x7f) | (this.buf[pos + 1] & 0x7f) << 7 | (this.buf[pos + 2] & 0x7f) << 14 | (this.buf[pos + 3]) << 21) + (this.buf[pos + 4] * 268435456); } else { this.skip(Protobuf.Varint); return 0; // throw new Error("TODO: Handle 6+ byte varints"); } }; Protobuf.prototype.readSVarint = function() { var num = this.readVarint(); if (num > 2147483647) throw new Error('TODO: Handle numbers >= 2^30'); // zigzag encoding return ((num >> 1) ^ -(num & 1)); }; Protobuf.prototype.readString = function() { var bytes = this.readVarint(); // TODO: bounds checking var chr = String.fromCharCode; var b = this.buf; var p = this.pos; var end = this.pos + bytes; var str = ''; while (p < end) { if (b[p] <= 0x7F) str += chr(b[p++]); else if (b[p] <= 0xBF) throw new Error('Invalid UTF-8 codepoint: ' + b[p]); else if (b[p] <= 0xDF) str += chr((b[p++] & 0x1F) << 6 | (b[p++] & 0x3F)); else if (b[p] <= 0xEF) str += chr((b[p++] & 0x1F) << 12 | (b[p++] & 0x3F) << 6 | (b[p++] & 0x3F)); else if (b[p] <= 0xF7) p += 4; // We can't handle these codepoints in JS, so skip. else if (b[p] <= 0xFB) p += 5; else if (b[p] <= 0xFD) p += 6; else throw new Error('Invalid UTF-8 codepoint: ' + b[p]); } this.pos += bytes; return str; }; Protobuf.prototype.readBuffer = function() { var bytes = this.readVarint(); var buffer = this.buf.subarray(this.pos, this.pos + bytes); this.pos += bytes; return buffer; }; Protobuf.prototype.readPacked = function(type) { // TODO: bounds checking var bytes = this.readVarint(); var end = this.pos + bytes; var array = []; while (this.pos < end) { array.push(this['read' + type]()); } return array; }; Protobuf.prototype.skip = function(val) { // TODO: bounds checking var type = val & 0x7; switch (type) { /* varint */ case Protobuf.Varint: while (this.buf[this.pos++] > 0x7f); break; /* 64 bit */ case Protobuf.Int64: this.pos += 8; break; /* length */ case Protobuf.Message: var bytes = this.readVarint(); this.pos += bytes; break; /* 32 bit */ case Protobuf.Int32: this.pos += 4; break; default: throw new Error('Unimplemented type: ' + type); } }; // === WRITING ================================================================= Protobuf.prototype.writeTag = function(tag, type) { this.writeVarint((tag << 3) | type); }; Protobuf.prototype.realloc = function(min) { var length = this.buf.length; while (length < this.pos + min) length *= 2; if (length != this.buf.length) { var buf = new Buffer(length); this.buf.copy(buf); this.buf = buf; } }; Protobuf.prototype.finish = function() { return this.buf.slice(0, this.pos); }; Protobuf.prototype.writePacked = function(type, tag, items) { if (!items.length) return; var message = new Protobuf(); for (var i = 0; i < items.length; i++) { message['write' + type](items[i]); } var data = message.finish(); this.writeTag(tag, Protobuf.Packed); this.writeBuffer(data); }; Protobuf.prototype.writeUInt32 = function(val) { this.realloc(4); this.buf.writeUInt32LE(val, this.pos); this.pos += 4; }; Protobuf.prototype.writeTaggedUInt32 = function(tag, val) { this.writeTag(tag, Protobuf.Int32); this.writeUInt32(val); }; Protobuf.prototype.writeVarint = function(val) { val = Number(val); if (isNaN(val)) { val = 0; } if (val <= 0x7f) { this.realloc(1); this.buf[this.pos++] = val; } else if (val <= 0x3fff) { this.realloc(2); this.buf[this.pos++] = 0x80 | ((val >>> 0) & 0x7f); this.buf[this.pos++] = 0x00 | ((val >>> 7) & 0x7f); } else if (val <= 0x1ffffff) { this.realloc(3); this.buf[this.pos++] = 0x80 | ((val >>> 0) & 0x7f); this.buf[this.pos++] = 0x80 | ((val >>> 7) & 0x7f); this.buf[this.pos++] = 0x00 | ((val >>> 14) & 0x7f); } else if (val <= 0xfffffff) { this.realloc(4); this.buf[this.pos++] = 0x80 | ((val >>> 0) & 0x7f); this.buf[this.pos++] = 0x80 | ((val >>> 7) & 0x7f); this.buf[this.pos++] = 0x80 | ((val >>> 14) & 0x7f); this.buf[this.pos++] = 0x00 | ((val >>> 21) & 0x7f); } else { while (val > 0) { var b = val & 0x7f; val = Math.floor(val / 128); if (val > 0) b |= 0x80 this.realloc(1); this.buf[this.pos++] = b; } } }; Protobuf.prototype.writeTaggedVarint = function(tag, val) { this.writeTag(tag, Protobuf.Varint); this.writeVarint(val); }; Protobuf.prototype.writeSVarint = function(val) { if (val >= 0) { this.writeVarint(val * 2); } else { this.writeVarint(val * -2 - 1); } }; Protobuf.prototype.writeTaggedSVarint = function(tag, val) { this.writeTag(tag, Protobuf.Varint); this.writeSVarint(val); }; Protobuf.prototype.writeBoolean = function(val) { this.writeVarint(Boolean(val)); }; Protobuf.prototype.writeTaggedBoolean = function(tag, val) { this.writeTaggedVarint(tag, Boolean(val)); }; Protobuf.prototype.writeString = function(str) { str = String(str); var bytes = Buffer.byteLength(str); this.writeVarint(bytes); this.realloc(bytes); this.buf.write(str, this.pos); this.pos += bytes; }; Protobuf.prototype.writeTaggedString = function(tag, str) { this.writeTag(tag, Protobuf.String); this.writeString(str); }; Protobuf.prototype.writeFloat = function(val) { this.realloc(4); this.buf.writeFloatLE(val, this.pos); this.pos += 4; }; Protobuf.prototype.writeTaggedFloat = function(tag, val) { this.writeTag(tag, Protobuf.Int32); this.writeFloat(val); }; Protobuf.prototype.writeDouble = function(val) { this.realloc(8); this.buf.writeDoubleLE(val, this.pos); this.pos += 8; }; Protobuf.prototype.writeTaggedDouble = function(tag, val) { this.writeTag(tag, Protobuf.Int64); this.writeDouble(val); }; Protobuf.prototype.writeBuffer = function(buffer) { var bytes = buffer.length; this.writeVarint(bytes); this.realloc(bytes); buffer.copy(this.buf, this.pos); this.pos += bytes; }; Protobuf.prototype.writeTaggedBuffer = function(tag, buffer) { this.writeTag(tag, Protobuf.String); this.writeBuffer(buffer); }; Protobuf.prototype.writeMessage = function(tag, protobuf) { var buffer = protobuf.finish(); this.writeTag(tag, Protobuf.Message); this.writeBuffer(buffer); }; }).call(this,require("buffer").Buffer) },{"buffer":88,"ieee754":97}],97:[function(require,module,exports){ module.exports=require(90) },{}],98:[function(require,module,exports){ 'use strict'; module.exports = Point; function Point(x, y) { this.x = x; this.y = y; } Point.prototype = { clone: function() { return new Point(this.x, this.y); }, add: function(p) { return this.clone()._add(p); }, sub: function(p) { return this.clone()._sub(p); }, mult: function(k) { return this.clone()._mult(k); }, div: function(k) { return this.clone()._div(k); }, rotate: function(a) { return this.clone()._rotate(a); }, matMult: function(m) { return this.clone()._matMult(m); }, unit: function() { return this.clone()._unit(); }, perp: function() { return this.clone()._perp(); }, round: function() { return this.clone()._round(); }, mag: function() { return Math.sqrt(this.x * this.x + this.y * this.y); }, equals: function(p) { return this.x === p.x && this.y === p.y; }, dist: function(p) { return Math.sqrt(this.distSqr(p)); }, distSqr: function(p) { var dx = p.x - this.x, dy = p.y - this.y; return dx * dx + dy * dy; }, angle: function() { return Math.atan2(this.y, this.x); }, angleTo: function(b) { return Math.atan2(this.y - b.y, this.x - b.x); }, angleWith: function(b) { return this.angleWithSep(b.x, b.y); }, // Find the angle of the two vectors, solving the formula for the cross product a x b = |a||b|sin(θ) for θ. angleWithSep: function(x, y) { return Math.atan2( this.x * y - this.y * x, this.x * x + this.y * y); }, _matMult: function(m) { var x = m[0] * this.x + m[1] * this.y, y = m[2] * this.x + m[3] * this.y; this.x = x; this.y = y; return this; }, _add: function(p) { this.x += p.x; this.y += p.y; return this; }, _sub: function(p) { this.x -= p.x; this.y -= p.y; return this; }, _mult: function(k) { this.x *= k; this.y *= k; return this; }, _div: function(k) { this.x /= k; this.y /= k; return this; }, _unit: function() { this._div(this.mag()); return this; }, _perp: function() { var y = this.y; this.y = this.x; this.x = -y; return this; }, _rotate: function(angle) { var cos = Math.cos(angle), sin = Math.sin(angle), x = cos * this.x - sin * this.y, y = sin * this.x + cos * this.y; this.x = x; this.y = y; return this; }, _round: function() { this.x = Math.round(this.x); this.y = Math.round(this.y); return this; } }; // constructs Point from an array if necessary Point.convert = function (a) { if (a instanceof Point) { return a; } if (Array.isArray(a)) { return new Point(a[0], a[1]); } return a; }; },{}],99:[function(require,module,exports){ (function() { var slice = [].slice; function queue(parallelism) { var q, tasks = [], started = 0, // number of tasks that have been started (and perhaps finished) active = 0, // number of tasks currently being executed (started but not finished) remaining = 0, // number of tasks not yet finished popping, // inside a synchronous task callback? error = null, await = noop, all; if (!parallelism) parallelism = Infinity; function pop() { while (popping = started < tasks.length && active < parallelism) { var i = started++, t = tasks[i], a = slice.call(t, 1); a.push(callback(i)); ++active; t[0].apply(null, a); } } function callback(i) { return function(e, r) { --active; if (error != null) return; if (e != null) { error = e; // ignore new tasks and squelch active callbacks started = remaining = NaN; // stop queued tasks from starting notify(); } else { tasks[i] = r; if (--remaining) popping || pop(); else notify(); } }; } function notify() { if (error != null) await(error); else if (all) await(error, tasks); else await.apply(null, [error].concat(tasks)); } return q = { defer: function() { if (!error) { tasks.push(arguments); ++remaining; pop(); } return q; }, await: function(f) { await = f; all = false; if (!remaining) notify(); return q; }, awaitAll: function(f) { await = f; all = true; if (!remaining) notify(); return q; } }; } function noop() {} queue.version = "1.0.7"; if (typeof define === "function" && define.amd) define(function() { return queue; }); else if (typeof module === "object" && module.exports) module.exports = queue; else this.queue = queue; })(); },{}],100:[function(require,module,exports){ /* (c) 2013, Vladimir Agafonkin RBush, a JavaScript library for high-performance 2D spatial indexing of points and rectangles. https://github.com/mourner/rbush */ (function () { 'use strict'; function rbush(maxEntries, format) { // jshint newcap: false, validthis: true if (!(this instanceof rbush)) { return new rbush(maxEntries, format); } // max entries in a node is 9 by default; min node fill is 40% for best performance this._maxEntries = Math.max(4, maxEntries || 9); this._minEntries = Math.max(2, Math.ceil(this._maxEntries * 0.4)); if (format) { this._initFormat(format); } this.clear(); } rbush.prototype = { all: function () { return this._all(this.data, []); }, search: function (bbox) { var node = this.data, result = []; if (!this._intersects(bbox, node.bbox)) { return result; } var nodesToSearch = [], i, len, child, childBBox; while (node) { for (i = 0, len = node.children.length; i < len; i++) { child = node.children[i]; childBBox = node.leaf ? this.toBBox(child) : child.bbox; if (this._intersects(bbox, childBBox)) { if (node.leaf) { result.push(child); } else if (this._contains(bbox, childBBox)) { this._all(child, result); } else { nodesToSearch.push(child); } } } node = nodesToSearch.pop(); } return result; }, load: function (data) { if (!(data && data.length)) { return this; } if (data.length < this._minEntries) { for (var i = 0, len = data.length; i < len; i++) { this.insert(data[i]); } return this; } // recursively build the tree with the given data from stratch using OMT algorithm var node = this._build(data.slice(), 0); if (!this.data.children.length) { // save as is if tree is empty this.data = node; } else if (this.data.height === node.height) { // split root if trees have the same height this._splitRoot(this.data, node); } else { if (this.data.height < node.height) { // swap trees if inserted one is bigger var tmpNode = this.data; this.data = node; node = tmpNode; } // insert the small tree into the large tree at appropriate level this._insert(node, this.data.height - node.height - 1, true); } return this; }, insert: function (item) { if (item) { this._insert(item, this.data.height - 1); } return this; }, clear: function () { this.data = { children: [], leaf: true, bbox: this._empty(), height: 1 }; return this; }, remove: function (item) { if (!item) { return this; } var node = this.data, bbox = this.toBBox(item), path = [], indexes = [], i, parent, index, goingUp; // depth-first iterative tree traversal while (node || path.length) { if (!node) { // go up node = path.pop(); parent = path[path.length - 1]; i = indexes.pop(); goingUp = true; } if (node.leaf) { // check current node index = node.children.indexOf(item); if (index !== -1) { // item found, remove the item and condense tree upwards node.children.splice(index, 1); path.push(node); this._condense(path); return this; } } if (!goingUp && !node.leaf && this._contains(node.bbox, bbox)) { // go down path.push(node); indexes.push(i); i = 0; parent = node; node = node.children[0]; } else if (parent) { // go right i++; node = parent.children[i]; goingUp = false; } else { // nothing found node = null; } } return this; }, toBBox: function (item) { return item; }, compareMinX: function (a, b) { return a[0] - b[0]; }, compareMinY: function (a, b) { return a[1] - b[1]; }, toJSON: function () { return this.data; }, fromJSON: function (data) { this.data = data; return this; }, _all: function (node, result) { var nodesToSearch = []; while (node) { if (node.leaf) { result.push.apply(result, node.children); } else { nodesToSearch.push.apply(nodesToSearch, node.children); } node = nodesToSearch.pop(); } return result; }, _build: function (items, level, height) { var N = items.length, M = this._maxEntries, node; if (N <= M) { node = { children: items, leaf: true, height: 1 }; this._calcBBox(node); return node; } if (!level) { // target height of the bulk-loaded tree height = Math.ceil(Math.log(N) / Math.log(M)); // target number of root entries to maximize storage utilization M = Math.ceil(N / Math.pow(M, height - 1)); items.sort(this.compareMinX); } // TODO eliminate recursion? node = { children: [], height: height }; var N1 = Math.ceil(N / M) * Math.ceil(Math.sqrt(M)), N2 = Math.ceil(N / M), compare = level % 2 === 1 ? this.compareMinX : this.compareMinY, i, j, slice, sliceLen, childNode; // split the items into M mostly square tiles for (i = 0; i < N; i += N1) { slice = items.slice(i, i + N1).sort(compare); for (j = 0, sliceLen = slice.length; j < sliceLen; j += N2) { // pack each entry recursively childNode = this._build(slice.slice(j, j + N2), level + 1, height - 1); node.children.push(childNode); } } this._calcBBox(node); return node; }, _chooseSubtree: function (bbox, node, level, path) { var i, len, child, targetNode, area, enlargement, minArea, minEnlargement; while (true) { path.push(node); if (node.leaf || path.length - 1 === level) { break; } minArea = minEnlargement = Infinity; for (i = 0, len = node.children.length; i < len; i++) { child = node.children[i]; area = this._area(child.bbox); enlargement = this._enlargedArea(bbox, child.bbox) - area; // choose entry with the least area enlargement if (enlargement < minEnlargement) { minEnlargement = enlargement; minArea = area < minArea ? area : minArea; targetNode = child; } else if (enlargement === minEnlargement) { // otherwise choose one with the smallest area if (area < minArea) { minArea = area; targetNode = child; } } } node = targetNode; } return node; }, _insert: function (item, level, isNode) { var bbox = isNode ? item.bbox : this.toBBox(item), insertPath = []; // find the best node for accommodating the item, saving all nodes along the path too var node = this._chooseSubtree(bbox, this.data, level, insertPath); // put the item into the node node.children.push(item); this._extend(node.bbox, bbox); // split on node overflow; propagate upwards if necessary while (level >= 0) { if (insertPath[level].children.length > this._maxEntries) { this._split(insertPath, level); level--; } else { break; } } // adjust bboxes along the insertion path this._adjustParentBBoxes(bbox, insertPath, level); }, // split overflowed node into two _split: function (insertPath, level) { var node = insertPath[level], M = node.children.length, m = this._minEntries; this._chooseSplitAxis(node, m, M); var newNode = { children: node.children.splice(this._chooseSplitIndex(node, m, M)), height: node.height }; if (node.leaf) { newNode.leaf = true; } this._calcBBox(node); this._calcBBox(newNode); if (level) { insertPath[level - 1].children.push(newNode); } else { this._splitRoot(node, newNode); } }, _splitRoot: function (node, newNode) { // split root node this.data = {}; this.data.children = [node, newNode]; this.data.height = node.height + 1; this._calcBBox(this.data); }, _chooseSplitIndex: function (node, m, M) { var i, bbox1, bbox2, overlap, area, minOverlap, minArea, index; minOverlap = minArea = Infinity; for (i = m; i <= M - m; i++) { bbox1 = this._distBBox(node, 0, i); bbox2 = this._distBBox(node, i, M); overlap = this._intersectionArea(bbox1, bbox2); area = this._area(bbox1) + this._area(bbox2); // choose distribution with minimum overlap if (overlap < minOverlap) { minOverlap = overlap; index = i; minArea = area < minArea ? area : minArea; } else if (overlap === minOverlap) { // otherwise choose distribution with minimum area if (area < minArea) { minArea = area; index = i; } } } return index; }, // sorts node children by the best axis for split _chooseSplitAxis: function (node, m, M) { var compareMinX = node.leaf ? this.compareMinX : this._compareNodeMinX, compareMinY = node.leaf ? this.compareMinY : this._compareNodeMinY, xMargin = this._allDistMargin(node, m, M, compareMinX), yMargin = this._allDistMargin(node, m, M, compareMinY); // if total distributions margin value is minimal for x, sort by minX, // otherwise it's already sorted by minY if (xMargin < yMargin) { node.children.sort(compareMinX); } }, // total margin of all possible split distributions where each node is at least m full _allDistMargin: function (node, m, M, compare) { node.children.sort(compare); var leftBBox = this._distBBox(node, 0, m), rightBBox = this._distBBox(node, M - m, M), margin = this._margin(leftBBox) + this._margin(rightBBox), i, child; for (i = m; i < M - m; i++) { child = node.children[i]; this._extend(leftBBox, node.leaf ? this.toBBox(child) : child.bbox); margin += this._margin(leftBBox); } for (i = M - m - 1; i >= m; i--) { child = node.children[i]; this._extend(rightBBox, node.leaf ? this.toBBox(child) : child.bbox); margin += this._margin(rightBBox); } return margin; }, // min bounding rectangle of node children from k to p-1 _distBBox: function (node, k, p) { var bbox = this._empty(); for (var i = k, child; i < p; i++) { child = node.children[i]; this._extend(bbox, node.leaf ? this.toBBox(child) : child.bbox); } return bbox; }, // calculate node's bbox from bboxes of its children _calcBBox: function (node) { node.bbox = this._distBBox(node, 0, node.children.length); }, _adjustParentBBoxes: function (bbox, path, level) { // adjust bboxes along the given tree path for (var i = level; i >= 0; i--) { this._extend(path[i].bbox, bbox); } }, _condense: function (path) { // go through the path, removing empty nodes and updating bboxes for (var i = path.length - 1, parent; i >= 0; i--) { if (path[i].children.length === 0) { if (i > 0) { parent = path[i - 1].children; parent.splice(parent.indexOf(path[i]), 1); } else { this.clear(); } } else { this._calcBBox(path[i]); } } }, _contains: function(a, b) { return a[0] <= b[0] && a[1] <= b[1] && b[2] <= a[2] && b[3] <= a[3]; }, _intersects: function (a, b) { return b[0] <= a[2] && b[1] <= a[3] && b[2] >= a[0] && b[3] >= a[1]; }, _extend: function (a, b) { a[0] = Math.min(a[0], b[0]); a[1] = Math.min(a[1], b[1]); a[2] = Math.max(a[2], b[2]); a[3] = Math.max(a[3], b[3]); return a; }, _area: function (a) { return (a[2] - a[0]) * (a[3] - a[1]); }, _margin: function (a) { return (a[2] - a[0]) + (a[3] - a[1]); }, _enlargedArea: function (a, b) { return (Math.max(b[2], a[2]) - Math.min(b[0], a[0])) * (Math.max(b[3], a[3]) - Math.min(b[1], a[1])); }, _intersectionArea: function (a, b) { var minX = Math.max(a[0], b[0]), minY = Math.max(a[1], b[1]), maxX = Math.min(a[2], b[2]), maxY = Math.min(a[3], b[3]); return Math.max(0, maxX - minX) * Math.max(0, maxY - minY); }, _empty: function () { return [Infinity, Infinity, -Infinity, -Infinity]; }, _compareNodeMinX: function (a, b) { return a.bbox[0] - b.bbox[0]; }, _compareNodeMinY: function (a, b) { return a.bbox[1] - b.bbox[1]; }, _initFormat: function (format) { // data format (minX, minY, maxX, maxY accessors) // uses eval-type function compilation instead of just accepting a toBBox function // because the algorithms are very sensitive to sorting functions performance, // so they should be dead simple and without inner calls // jshint evil: true var compareArr = ['return a', ' - b', ';']; this.compareMinX = new Function('a', 'b', compareArr.join(format[0])); this.compareMinY = new Function('a', 'b', compareArr.join(format[1])); this.toBBox = new Function('a', 'return [a' + format.join(', a') + '];'); } }; if (typeof define === 'function' && define.amd) { define(function() { return rbush; }); } else if (typeof module !== 'undefined') { module.exports = rbush; } else if (typeof self !== 'undefined') { self.rbush = rbush; } else { window.rbush = rbush; } })(); },{}],101:[function(require,module,exports){ /* * Copyright (C) 2008 Apple Inc. All Rights Reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * * Ported from Webkit * http://svn.webkit.org/repository/webkit/trunk/Source/WebCore/platform/graphics/UnitBezier.h */ module.exports = UnitBezier; function UnitBezier(p1x, p1y, p2x, p2y) { // Calculate the polynomial coefficients, implicit first and last control points are (0,0) and (1,1). this.cx = 3.0 * p1x; this.bx = 3.0 * (p2x - p1x) - this.cx; this.ax = 1.0 - this.cx - this.bx; this.cy = 3.0 * p1y; this.by = 3.0 * (p2y - p1y) - this.cy; this.ay = 1.0 - this.cy - this.by; this.p1x = p1x; this.p1y = p2y; this.p2x = p2x; this.p2y = p2y; } UnitBezier.prototype.sampleCurveX = function(t) { // `ax t^3 + bx t^2 + cx t' expanded using Horner's rule. return ((this.ax * t + this.bx) * t + this.cx) * t; }; UnitBezier.prototype.sampleCurveY = function(t) { return ((this.ay * t + this.by) * t + this.cy) * t; }; UnitBezier.prototype.sampleCurveDerivativeX = function(t) { return (3.0 * this.ax * t + 2.0 * this.bx) * t + this.cx; }; UnitBezier.prototype.solveCurveX = function(x, epsilon) { if (typeof epsilon === 'undefined') epsilon = 1e-6; var t0, t1, t2, x2, i; // First try a few iterations of Newton's method -- normally very fast. for (t2 = x, i = 0; i < 8; i++) { x2 = this.sampleCurveX(t2) - x; if (Math.abs(x2) < epsilon) return t2; var d2 = this.sampleCurveDerivativeX(t2); if (Math.abs(d2) < 1e-6) break; t2 = t2 - x2 / d2; } // Fall back to the bisection method for reliability. t0 = 0.0; t1 = 1.0; t2 = x; if (t2 < t0) return t0; if (t2 > t1) return t1; while (t0 < t1) { x2 = this.sampleCurveX(t2); if (Math.abs(x2 - x) < epsilon) return t2; if (x > x2) { t0 = t2; } else { t1 = t2; } t2 = (t1 - t0) * 0.5 + t0; } // Failure. return t2; }; UnitBezier.prototype.solve = function(x, epsilon) { return this.sampleCurveY(this.solveCurveX(x, epsilon)); }; },{}],102:[function(require,module,exports){ module.exports.VectorTile = require('./lib/vectortile.js'); module.exports.VectorTileFeature = require('./lib/vectortilefeature.js'); module.exports.VectorTileLayer = require('./lib/vectortilelayer.js'); },{"./lib/vectortile.js":103,"./lib/vectortilefeature.js":104,"./lib/vectortilelayer.js":105}],103:[function(require,module,exports){ 'use strict'; var VectorTileLayer = require('./vectortilelayer'); module.exports = VectorTile; function VectorTile(buffer, end) { this.layers = {}; this._buffer = buffer; end = end || buffer.length; while (buffer.pos < end) { var val = buffer.readVarint(), tag = val >> 3; if (tag == 3) { var layer = this.readLayer(); if (layer.length) this.layers[layer.name] = layer; } else { buffer.skip(val); } } } VectorTile.prototype.readLayer = function() { var buffer = this._buffer, bytes = buffer.readVarint(), end = buffer.pos + bytes, layer = new VectorTileLayer(buffer, end); buffer.pos = end; return layer; }; },{"./vectortilelayer":105}],104:[function(require,module,exports){ 'use strict'; var Point = require('point-geometry'); module.exports = VectorTileFeature; function VectorTileFeature(buffer, end, extent, keys, values) { this.properties = {}; // Public this.extent = extent; this.type = 0; // Private this._buffer = buffer; this._geometry = -1; end = end || buffer.length; while (buffer.pos < end) { var val = buffer.readVarint(), tag = val >> 3; if (tag == 1) { this._id = buffer.readVarint(); } else if (tag == 2) { var tagEnd = buffer.pos + buffer.readVarint(); while (buffer.pos < tagEnd) { var key = keys[buffer.readVarint()]; var value = values[buffer.readVarint()]; this.properties[key] = value; } } else if (tag == 3) { this.type = buffer.readVarint(); } else if (tag == 4) { this._geometry = buffer.pos; buffer.skip(val); } else { buffer.skip(val); } } } VectorTileFeature.types = ['Unknown', 'Point', 'LineString', 'Polygon']; VectorTileFeature.prototype.loadGeometry = function() { var buffer = this._buffer; buffer.pos = this._geometry; var bytes = buffer.readVarint(), end = buffer.pos + bytes, cmd = 1, length = 0, x = 0, y = 0, lines = [], line; while (buffer.pos < end) { if (!length) { var cmd_length = buffer.readVarint(); cmd = cmd_length & 0x7; length = cmd_length >> 3; } length--; if (cmd === 1 || cmd === 2) { x += buffer.readSVarint(); y += buffer.readSVarint(); if (cmd === 1) { // moveTo if (line) { lines.push(line); } line = []; } line.push(new Point(x, y)); } else if (cmd === 7) { // closePolygon line.push(line[0].clone()); } else { throw new Error('unknown command ' + cmd); } } if (line) lines.push(line); return lines; }; VectorTileFeature.prototype.bbox = function() { var buffer = this._buffer; buffer.pos = this._geometry; var bytes = buffer.readVarint(), end = buffer.pos + bytes, cmd = 1, length = 0, x = 0, y = 0, x1 = Infinity, x2 = -Infinity, y1 = Infinity, y2 = -Infinity; while (buffer.pos < end) { if (!length) { var cmd_length = buffer.readVarint(); cmd = cmd_length & 0x7; length = cmd_length >> 3; } length--; if (cmd === 1 || cmd === 2) { x += buffer.readSVarint(); y += buffer.readSVarint(); if (x < x1) x1 = x; if (x > x2) x2 = x; if (y < y1) y1 = y; if (y > y2) y2 = y; } else if (cmd !== 7) { throw new Error('unknown command ' + cmd); } } return [x1, y1, x2, y2]; }; },{"point-geometry":98}],105:[function(require,module,exports){ 'use strict'; var VectorTileFeature = require('./vectortilefeature.js'); module.exports = VectorTileLayer; function VectorTileLayer(buffer, end) { // Public this.version = 1; this.name = null; this.extent = 4096; this.length = 0; // Private this._buffer = buffer; this._keys = []; this._values = []; this._features = []; var val, tag; end = end || buffer.length; while (buffer.pos < end) { val = buffer.readVarint(); tag = val >> 3; if (tag === 15) { this.version = buffer.readVarint(); } else if (tag === 1) { this.name = buffer.readString(); } else if (tag === 5) { this.extent = buffer.readVarint(); } else if (tag === 2) { this.length++; this._features.push(buffer.pos); buffer.skip(val); } else if (tag === 3) { this._keys.push(buffer.readString()); } else if (tag === 4) { this._values.push(this.readFeatureValue()); } else { buffer.skip(val); } } } VectorTileLayer.prototype.readFeatureValue = function() { var buffer = this._buffer, value = null, bytes = buffer.readVarint(), end = buffer.pos + bytes, val, tag; while (buffer.pos < end) { val = buffer.readVarint(); tag = val >> 3; if (tag == 1) { value = buffer.readString(); } else if (tag == 2) { throw new Error('read float'); } else if (tag == 3) { value = buffer.readDouble(); } else if (tag == 4) { value = buffer.readVarint(); } else if (tag == 5) { throw new Error('read uint'); } else if (tag == 6) { value = buffer.readSVarint(); } else if (tag == 7) { value = Boolean(buffer.readVarint()); } else { buffer.skip(val); } } return value; }; // return feature `i` from this layer as a `VectorTileFeature` VectorTileLayer.prototype.feature = function(i) { if (i < 0 || i >= this._features.length) throw new Error('feature index out of bounds'); this._buffer.pos = this._features[i]; var end = this._buffer.readVarint() + this._buffer.pos; return new VectorTileFeature(this._buffer, end, this.extent, this._keys, this._values); }; },{"./vectortilefeature.js":104}]},{},[22]) //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/browserify/node_modules/browser-pack/_prelude.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/data/buffer/buffer.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/data/buffer/bufferset.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/data/buffer/fillelementsbuffer.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/data/buffer/fillvertexbuffer.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/data/buffer/glyphvertexbuffer.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/data/buffer/iconvertexbuffer.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/data/buffer/lineelementbuffer.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/data/buffer/linevertexbuffer.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/data/buffer/outlineelementsbuffer.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/data/createbucket.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/data/elementgroups.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/data/featuretree.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/data/fillbucket.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/data/linebucket.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/data/rasterbucket.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/data/symbolbucket.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/geo/latlng.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/geo/latlngbounds.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/geo/transform.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/lib/debugtext.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/lib/glmatrix.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/mapbox-gl.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/render/drawbackground.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/render/drawcomposited.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/render/drawdebug.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/render/drawfill.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/render/drawline.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/render/drawraster.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/render/drawsymbol.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/render/drawvertices.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/render/framehistory.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/render/glutil.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/render/painter.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/render/prerendered.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/render/shaders.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/source/geojsonsource.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/source/geojsontile.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/source/geojsonwrapper.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/source/rastertile.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/source/source.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/source/tile.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/source/tilecoord.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/source/tilegeojson.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/source/vectortile.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/source/videosource.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/source/worker.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/source/workertile.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/style/animationloop.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/style/bucketfilter.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/style/calculatedstyle.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/style/imagesprite.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/style/renderproperties.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/style/style.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/style/styleconstant.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/style/styledeclaration.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/style/styletransition.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/symbol/anchor.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/symbol/binpack.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/symbol/collision.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/symbol/glyphatlas.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/symbol/glyphsource.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/symbol/interpolate.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/symbol/placement.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/symbol/resolvetext.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/symbol/rotationrange.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/symbol/shaping.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/ui/control/attribution.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/ui/control/control.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/ui/control/navigation.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/ui/easings.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/ui/handlers.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/ui/hash.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/ui/interaction.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/ui/map.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/util/actor.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/util/browser/ajax.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/util/browser/browser.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/util/browser/canvas.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/util/browser/dispatcher.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/util/config.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/util/dom.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/util/evented.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/util/glyphs.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/util/mrucache.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/util/token.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/util/url.js","/Users/ndoiron404/Documents/mapbox-gl-js/js/util/util.js","/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/browserify/node_modules/buffer/index.js","/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/browserify/node_modules/buffer/node_modules/base64-js/lib/b64.js","/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/browserify/node_modules/buffer/node_modules/ieee754/index.js","/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/csscolorparser/csscolorparser.js","/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/geojson-rewind/index.js","/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/geojson-rewind/node_modules/geojson-area/index.js","/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/geojson-rewind/node_modules/geojson-area/node_modules/wgs84/index.js","/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/mapbox-gl-style-spec/reference/v4.json","/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/pbf/index.js","/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/point-geometry/index.js","/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/queue-async/queue.js","/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/rbush/rbush.js","/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/unitbezier/index.js","/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/vector-tile/index.js","/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/vector-tile/lib/vectortile.js","/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/vector-tile/lib/vectortilefeature.js","/Users/ndoiron404/Documents/mapbox-gl-js/node_modules/vector-tile/lib/vectortilelayer.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9PA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChUA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3YA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpeA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/QA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7XA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/TA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3QA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChWA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5NA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChcA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjpCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChEA;AACA;AACA;AACA;;ACHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC33BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AClSA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxhBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzGA;AACA;AACA;AACA;;ACHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","'use strict';\n\n// a simple wrapper around a single arraybuffer\n\nmodule.exports = Buffer;\n\nfunction Buffer(buffer) {\n    if (!buffer) {\n        this.array = new ArrayBuffer(this.defaultLength);\n        this.length = this.defaultLength;\n        this.setupViews();\n\n    } else {\n        // we only recreate buffers after receiving them from workers for binding to gl,\n        // so we only need these 2 properties\n        this.array = buffer.array;\n        this.pos = buffer.pos;\n    }\n}\n\nBuffer.prototype = {\n    pos: 0,\n    itemSize: 4, // bytes in one item\n    defaultLength: 8192, // initial buffer size\n    arrayType: 'ARRAY_BUFFER', // gl buffer type\n\n    get index() {\n        return this.pos / this.itemSize;\n    },\n\n    setupViews: function() {\n        // set up views for each type to add data of different types to the same buffer\n        this.ubytes = new Uint8Array(this.array);\n        this.bytes = new Int8Array(this.array);\n        this.ushorts = new Uint16Array(this.array);\n        this.shorts = new Int16Array(this.array);\n    },\n\n    // binds the buffer to a webgl context\n    bind: function(gl) {\n        var type = gl[this.arrayType];\n        if (!this.buffer) {\n            this.buffer = gl.createBuffer();\n            gl.bindBuffer(type, this.buffer);\n            gl.bufferData(type, new DataView(this.array, 0, this.pos), gl.STATIC_DRAW);\n\n            // dump array buffer once it's bound to gl\n            this.array = null;\n        } else {\n            gl.bindBuffer(type, this.buffer);\n        }\n    },\n\n    destroy: function(gl) {\n        if (this.buffer) {\n            gl.deleteBuffer(this.buffer);\n        }\n    },\n\n    // increase the buffer size by 50% if a new item doesn't fit\n    resize: function() {\n        if (this.length < this.pos + this.itemSize) {\n\n            while (this.length < this.pos + this.itemSize) {\n                // increase the length by 50% but keep it even\n                this.length = Math.round(this.length * 1.5 / 2) * 2;\n            }\n\n            // array buffers can't be resized, so we create a new one and reset all bytes there\n            this.array = new ArrayBuffer(this.length);\n\n            var ubytes = new Uint8Array(this.array);\n            ubytes.set(this.ubytes);\n\n            this.setupViews();\n        }\n    }\n};\n","'use strict';\n\nvar LineVertexBuffer = require('./linevertexbuffer.js');\nvar LineElementBuffer = require('./lineelementbuffer.js');\nvar FillVertexBuffer = require('./fillvertexbuffer.js');\nvar FillElementBuffer = require('./fillelementsbuffer.js');\nvar OutlineElementBuffer = require('./outlineelementsbuffer.js');\nvar GlyphVertexBuffer = require('./glyphvertexbuffer.js');\nvar IconVertexBuffer = require('./iconvertexbuffer.js');\n\nmodule.exports = function(bufferset) {\n    bufferset = bufferset || {};\n    return {\n        glyphVertex: new GlyphVertexBuffer(bufferset.glyphVertex),\n        iconVertex: new IconVertexBuffer(bufferset.iconVertex),\n        fillVertex: new FillVertexBuffer(bufferset.fillVertex),\n        fillElement: new FillElementBuffer(bufferset.fillElement),\n        outlineElement: new OutlineElementBuffer(bufferset.outlineElement),\n        lineVertex: new LineVertexBuffer(bufferset.lineVertex),\n        lineElement: new LineElementBuffer(bufferset.lineElement)\n    };\n};\n","'use strict';\n\nvar Buffer = require('./buffer.js');\n\nmodule.exports = FillElementsBuffer;\n\nfunction FillElementsBuffer(buffer) {\n    Buffer.call(this, buffer);\n}\n\nFillElementsBuffer.prototype = Object.create(Buffer.prototype);\n\nFillElementsBuffer.prototype.itemSize = 6; // bytes per triangle (3 * unsigned short == 6 bytes)\nFillElementsBuffer.prototype.arrayType = 'ELEMENT_ARRAY_BUFFER';\n\nFillElementsBuffer.prototype.add = function(a, b, c) {\n    var pos2 = this.pos / 2;\n\n    this.resize();\n\n    this.ushorts[pos2 + 0] = a;\n    this.ushorts[pos2 + 1] = b;\n    this.ushorts[pos2 + 2] = c;\n\n    this.pos += this.itemSize;\n};\n","'use strict';\n\nvar Buffer = require('./buffer.js');\n\nmodule.exports = FillVertexBuffer;\n\nfunction FillVertexBuffer(buffer) {\n    Buffer.call(this, buffer);\n}\n\nFillVertexBuffer.prototype = Object.create(Buffer.prototype);\n\nFillVertexBuffer.prototype.itemSize = 4; // bytes per vertex (2 * short == 4 bytes)\n\nFillVertexBuffer.prototype.add = function(x, y) {\n    var pos2 = this.pos / 2;\n\n    this.resize();\n\n    this.shorts[pos2 + 0] = x;\n    this.shorts[pos2 + 1] = y;\n\n    this.pos += this.itemSize;\n};\n","'use strict';\n\nvar Buffer = require('./buffer.js');\n\nmodule.exports = GlyphVertexBuffer;\n\nfunction GlyphVertexBuffer(buffer) {\n    Buffer.call(this, buffer);\n}\n\nGlyphVertexBuffer.prototype = Object.create(Buffer.prototype);\n\nGlyphVertexBuffer.prototype.defaultLength = 2048 * 16;\nGlyphVertexBuffer.prototype.itemSize = 16;\n\n// Converts the 0..2pi to an int16 range\nGlyphVertexBuffer.angleFactor = 128 / Math.PI;\n\nGlyphVertexBuffer.prototype.add = function(x, y, ox, oy, tx, ty, angle, minzoom, range, maxzoom, labelminzoom) {\n    var pos = this.pos,\n        pos2 = pos / 2,\n        angleFactor = GlyphVertexBuffer.angleFactor;\n\n    this.resize();\n\n    this.shorts[pos2 + 0] = x;\n    this.shorts[pos2 + 1] = y;\n    this.shorts[pos2 + 2] = Math.round(ox * 64); // use 1/64 pixels for placement\n    this.shorts[pos2 + 3] = Math.round(oy * 64);\n\n    this.ubytes[pos + 8] = Math.floor((labelminzoom || 0) * 10);\n    this.ubytes[pos + 9] = Math.floor((minzoom || 0) * 10); // 1/10 zoom levels: z16 == 160.\n    this.ubytes[pos + 10] = Math.floor(Math.min(maxzoom || 25, 25) * 10); // 1/10 zoom levels: z16 == 160.\n    this.ubytes[pos + 11] = Math.round(angle * angleFactor) % 256;\n    this.ubytes[pos + 12] = Math.max(Math.round(range[0] * angleFactor), 0) % 256;\n    this.ubytes[pos + 13] = Math.min(Math.round(range[1] * angleFactor), 255) % 256;\n\n    this.ubytes[pos + 14] = Math.floor(tx / 4);\n    this.ubytes[pos + 15] = Math.floor(ty / 4);\n\n    this.pos += this.itemSize;\n};\n","'use strict';\n\nvar Buffer = require('./buffer.js');\n\nmodule.exports = GlyphVertexBuffer;\n\nfunction GlyphVertexBuffer(buffer) {\n    Buffer.call(this, buffer);\n}\n\nGlyphVertexBuffer.prototype = Object.create(Buffer.prototype);\n\nGlyphVertexBuffer.prototype.defaultLength = 2048 * 20;\nGlyphVertexBuffer.prototype.itemSize = 20;\n\n// Converts the 0..2pi to an int16 range\nGlyphVertexBuffer.angleFactor = 128 / Math.PI;\n\nGlyphVertexBuffer.prototype.add = function(x, y, ox, oy, tx, ty, angle, minzoom, range, maxzoom, labelminzoom) {\n    var pos = this.pos,\n        pos2 = pos / 2,\n        angleFactor = GlyphVertexBuffer.angleFactor;\n\n    this.resize();\n\n    this.shorts[pos2 + 0] = x;\n    this.shorts[pos2 + 1] = y;\n    this.shorts[pos2 + 2] = Math.round(ox * 64); // use 1/64 pixels for placement\n    this.shorts[pos2 + 3] = Math.round(oy * 64);\n\n    this.ubytes[pos + 8] = Math.floor((labelminzoom || 0) * 10);\n    this.ubytes[pos + 9] = Math.floor((minzoom || 0) * 10); // 1/10 zoom levels: z16 == 160.\n    this.ubytes[pos + 10] = Math.floor(Math.min(maxzoom || 25, 25) * 10); // 1/10 zoom levels: z16 == 160.\n    this.ubytes[pos + 11] = Math.round(angle * angleFactor) % 256;\n    this.ubytes[pos + 12] = Math.max(Math.round(range[0] * angleFactor), 0) % 256;\n    this.ubytes[pos + 13] = Math.min(Math.round(range[1] * angleFactor), 255) % 256;\n\n    this.shorts[pos2 + 8] = tx;\n    this.shorts[pos2 + 9] = ty;\n\n    this.pos += this.itemSize;\n};\n","'use strict';\n\nvar Buffer = require('./buffer.js');\n\nmodule.exports = LineElementBuffer;\n\nfunction LineElementBuffer(buffer) {\n    Buffer.call(this, buffer);\n}\n\nLineElementBuffer.prototype = Object.create(Buffer.prototype);\n\nLineElementBuffer.prototype.itemSize = 6; // bytes per triangle (3 * unsigned short == 6 bytes)\nLineElementBuffer.prototype.arrayType = 'ELEMENT_ARRAY_BUFFER';\n\nLineElementBuffer.prototype.add = function(a, b, c) {\n    var pos2 = this.pos / 2;\n\n    this.resize();\n\n    this.ushorts[pos2 + 0] = a;\n    this.ushorts[pos2 + 1] = b;\n    this.ushorts[pos2 + 2] = c;\n\n    this.pos += this.itemSize;\n};\n","'use strict';\n\nvar Buffer = require('./buffer.js');\n\nmodule.exports = LineVertexBuffer;\n\nfunction LineVertexBuffer(buffer) {\n    Buffer.call(this, buffer);\n}\n\n// scale the extrusion vector so that the normal length is this value.\n// contains the \"texture\" normals (-1..1). this is distinct from the extrude\n// normals for line joins, because the x-value remains 0 for the texture\n// normal array, while the extrude normal actually moves the vertex to create\n// the acute/bevelled line join.\nLineVertexBuffer.extrudeScale = 63;\n\nLineVertexBuffer.prototype = Object.create(Buffer.prototype);\n\nLineVertexBuffer.prototype.itemSize = 8; // bytes per vertex (2 * short + 1 * short + 2 * byte = 8 bytes)\nLineVertexBuffer.prototype.defaultLength = 32768;\n\n// add a vertex to this buffer;\n// x, y - vertex position\n// ex, ey - extrude normal\n// tx, ty - texture normal\n\nLineVertexBuffer.prototype.add = function(point, extrude, tx, ty, linesofar) {\n    var pos = this.pos,\n        pos2 = pos / 2,\n        index = this.index,\n        extrudeScale = LineVertexBuffer.extrudeScale;\n\n    this.resize();\n\n    this.shorts[pos2 + 0] = (Math.floor(point.x) * 2) | tx;\n    this.shorts[pos2 + 1] = (Math.floor(point.y) * 2) | ty;\n    this.shorts[pos2 + 2] = Math.round(linesofar || 0);\n    this.bytes[pos + 6] = Math.round(extrudeScale * extrude.x);\n    this.bytes[pos + 7] = Math.round(extrudeScale * extrude.y);\n\n    this.pos += this.itemSize;\n    return index;\n};\n","'use strict';\n\nvar Buffer = require('./buffer.js');\n\nmodule.exports = OutlineElementsBuffer;\n\nfunction OutlineElementsBuffer(buffer) {\n    Buffer.call(this, buffer);\n}\n\nOutlineElementsBuffer.prototype = Object.create(Buffer.prototype);\n\nOutlineElementsBuffer.prototype.itemSize = 4; // bytes per line (2 * unsigned short == 4 bytes)\nOutlineElementsBuffer.prototype.arrayType = 'ELEMENT_ARRAY_BUFFER';\n\nOutlineElementsBuffer.prototype.add = function(a, b) {\n    var pos2 = this.pos / 2;\n\n    this.resize();\n\n    this.ushorts[pos2 + 0] = a;\n    this.ushorts[pos2 + 1] = b;\n\n    this.pos += this.itemSize;\n};\n","'use strict';\n\nmodule.exports = createBucket;\n\nvar LineBucket = require('./linebucket.js');\nvar FillBucket = require('./fillbucket.js');\nvar SymbolBucket = require('./symbolbucket.js');\nvar RasterBucket = require('./rasterbucket.js');\nvar RenderProperties = require('../style/renderproperties.js');\n\nfunction createBucket(layer, buffers, collision, indices) {\n\n    if (!RenderProperties[layer.type]) {\n        //console.warn('unknown bucket type');\n        return;\n    }\n\n    var info = new RenderProperties[layer.type](layer.render);\n\n    var BucketClass =\n        layer.type === 'line' ? LineBucket :\n        layer.type === 'fill' ? FillBucket :\n        layer.type === 'symbol' ? SymbolBucket :\n        layer.type === 'raster' ? RasterBucket : null;\n\n    var bucket = new BucketClass(info, buffers, collision, indices);\n    bucket.type = layer.type;\n    bucket.interactive = layer.interactive;\n    bucket.minZoom = layer['min-zoom'];\n    bucket.maxZoom = layer['max-zoom'];\n\n    return bucket;\n}\n","'use strict';\n\nmodule.exports = ElementGroups;\n\nfunction ElementGroups(vertexBuffer, elementBuffer, secondElementBuffer) {\n\n    this.vertexBuffer = vertexBuffer;\n    this.elementBuffer = elementBuffer;\n    this.secondElementBuffer = secondElementBuffer;\n    this.groups = [];\n}\n\nElementGroups.prototype.makeRoomFor = function(numVertices) {\n    if (!this.current || this.current.vertexLength + numVertices > 65535) {\n        this.current = new ElementGroup(this.vertexBuffer.index,\n                this.elementBuffer && this.elementBuffer.index,\n                this.secondElementBuffer && this.secondElementBuffer.index);\n        this.groups.push(this.current);\n    }\n};\n\nfunction ElementGroup(vertexStartIndex, elementStartIndex, secondElementStartIndex)  {\n    // the offset into the vertex buffer of the first vertex in this group\n    this.vertexStartIndex = vertexStartIndex;\n    this.elementStartIndex = elementStartIndex;\n    this.secondElementStartIndex = secondElementStartIndex;\n    this.elementLength = 0;\n    this.vertexLength = 0;\n    this.secondElementLength = 0;\n}\n","'use strict';\n\nvar rbush = require('rbush'),\n    Point = require('point-geometry');\n\nmodule.exports = FeatureTree;\n\nfunction FeatureTree(getGeometry, getType) {\n\n    this.getGeometry = getGeometry;\n    this.getType = getType;\n\n    this.rtree = rbush(9);\n    this.toBeInserted = [];\n}\n\nFeatureTree.prototype.insert = function(bbox, bucket_name, feature) {\n    bbox.bucket = bucket_name;\n    bbox.feature = feature;\n    this.toBeInserted.push(bbox);\n};\n\n// bulk insert into tree\nFeatureTree.prototype._load = function() {\n    this.rtree.load(this.toBeInserted);\n    this.toBeInserted = [];\n};\n\n// Finds features in this tile at a particular position.\nFeatureTree.prototype.query = function(args, callback) {\n\n    if (this.toBeInserted.length) this._load();\n\n    var radius = args.params && args.params.radius || 0;\n    radius *= 4096 / args.scale;\n\n    var x = args.x,\n        y = args.y;\n\n    var matching = this.rtree.search([ x - radius, y - radius, x + radius, y + radius ]);\n\n    if (args.params.buckets) {\n        this.queryBuckets(matching, x, y, radius, args.params, callback);\n    } else {\n        this.queryFeatures(matching, x, y, radius, args.params, callback);\n    }\n};\n\nFeatureTree.prototype.queryFeatures = function(matching, x, y, radius, params, callback) {\n    var result = [];\n    for (var i = 0; i < matching.length; i++) {\n        var feature = matching[i].feature;\n        var type = this.getType(feature);\n        var geometry = this.getGeometry(feature);\n\n\n        if (params.bucket && matching[i].bucket !== params.bucket) continue;\n        if (params.type && type !== params.type) continue;\n\n        if (geometryContainsPoint(geometry, type, new Point(x, y), radius)) {\n            var props = {\n                _bucket: matching[i].bucket,\n                _type: type\n            };\n\n            if (params.geometry) {\n                props._geometry = geometry;\n            }\n\n            for (var key in feature) {\n                if (feature.hasOwnProperty(key) && key[0] !== '_') {\n                    props[key] = feature[key];\n                }\n            }\n            result.push(props);\n        }\n    }\n\n    callback(null, result);\n};\n\n// Lists all buckets that at the position.\nFeatureTree.prototype.queryBuckets = function(matching, x, y, radius, params, callback) {\n    var buckets = [];\n    for (var i = 0; i < matching.length; i++) {\n        if (buckets.indexOf(matching[i].bucket) >= 0) continue;\n\n        var feature = matching[i].feature;\n        var type = this.getType(feature);\n        var geometry = this.getGeometry(feature);\n        if (geometryContainsPoint(geometry, type, new Point(x, y), radius)) {\n            buckets.push(matching[i].bucket);\n        }\n    }\n\n    callback(null, buckets);\n};\n\n\nfunction geometryContainsPoint(rings, type, p, radius) {\n    if (type === 'Point') {\n        return pointContainsPoint(rings, p, radius);\n    } else if (type === 'LineString') {\n        return lineContainsPoint(rings, p, radius);\n    } else if (type === 'Polygon') {\n        return polyContainsPoint(rings, p) ? true : lineContainsPoint(rings, p, radius);\n    } else {\n        return false;\n    }\n}\n\n// Code from http://stackoverflow.com/a/1501725/331379.\nfunction distToSegmentSquared(p, v, w) {\n    var l2 = v.distSqr(w);\n    if (l2 === 0) return p.distSqr(v);\n    var t = ((p.x - v.x) * (w.x - v.x) + (p.y - v.y) * (w.y - v.y)) / l2;\n    if (t < 0) return p.distSqr(v);\n    if (t > 1) return p.distSqr(w);\n    return p.distSqr(w.sub(v)._mult(t)._add(v));\n}\n\nfunction lineContainsPoint(rings, p, radius) {\n    var r = radius * radius;\n\n    for (var i = 0; i < rings.length; i++) {\n        var ring = rings[i];\n        for (var j = 1; j < ring.length; j++) {\n            // Find line segments that have a distance <= radius^2 to p\n            // In that case, we treat the line as \"containing point p\".\n            var v = ring[j-1], w = ring[j];\n            if (distToSegmentSquared(p, v, w) < r) return true;\n        }\n    }\n    return false;\n}\n\n// point in polygon ray casting algorithm\nfunction polyContainsPoint(rings, p) {\n    var c = false,\n        ring, p1, p2;\n\n    for (var k = 0; k < rings.length; k++) {\n        ring = rings[k];\n        for (var i = 0, j = ring.length - 1; i < ring.length; j = i++) {\n            p1 = ring[i];\n            p2 = ring[j];\n            if (((p1.y > p.y) != (p2.y > p.y)) && (p.x < (p2.x - p1.x) * (p.y - p1.y) / (p2.y - p1.y) + p1.x)) {\n                c = !c;\n            }\n        }\n    }\n    return c;\n}\n\nfunction pointContainsPoint(rings, p, radius) {\n    var r = radius * radius;\n\n    for (var i = 0; i < rings.length; i++) {\n        var ring = rings[i];\n        for (var j = 0; j < ring.length; j++) {\n            if (ring[j].distSqr(p) <= r) return true;\n        }\n    }\n    return false;\n}\n\n\n","'use strict';\n\nvar ElementGroups = require('./elementgroups.js');\n\nmodule.exports = FillBucket;\n\nfunction FillBucket(info, buffers, placement, elementGroups) {\n    this.info = info;\n    this.buffers = buffers;\n    this.elementGroups = elementGroups || new ElementGroups(buffers.fillVertex, buffers.fillElement, buffers.outlineElement);\n}\n\nFillBucket.prototype.addFeatures = function() {\n    var features = this.features;\n    for (var i = 0; i < features.length; i++) {\n        var feature = features[i];\n        this.addFeature(feature.loadGeometry());\n    }\n};\n\nFillBucket.prototype.addFeature = function(lines) {\n    for (var i = 0; i < lines.length; i++) {\n        this.addFill(lines[i]);\n    }\n};\n\nFillBucket.prototype.addFill = function(vertices) {\n    if (vertices.length < 3) {\n        //console.warn('a fill must have at least three vertices');\n        return;\n    }\n\n    // Calculate the total number of vertices we're going to produce so that we\n    // can resize the buffer beforehand, or detect whether the current line\n    // won't fit into the buffer anymore.\n    // In order to be able to use the vertex buffer for drawing the antialiased\n    // outlines, we separate all polygon vertices with a degenerate (out-of-\n    // viewplane) vertex.\n\n    var len = vertices.length;\n\n    // Check whether this geometry buffer can hold all the required vertices.\n    this.elementGroups.makeRoomFor(len + 1);\n    var elementGroup = this.elementGroups.current;\n\n    var fillVertex = this.buffers.fillVertex;\n    var fillElement = this.buffers.fillElement;\n    var outlineElement = this.buffers.outlineElement;\n\n    // Start all lines with a degenerate vertex\n    elementGroup.vertexLength++;\n\n    // We're generating triangle fans, so we always start with the first coordinate in this polygon.\n    var firstIndex = fillVertex.index - elementGroup.vertexStartIndex,\n        prevIndex, currentIndex, currentVertex;\n\n    for (var i = 0; i < vertices.length; i++) {\n        currentIndex = fillVertex.index - elementGroup.vertexStartIndex;\n        currentVertex = vertices[i];\n\n        fillVertex.add(currentVertex.x, currentVertex.y);\n        elementGroup.vertexLength++;\n\n        // Only add triangles that have distinct vertices.\n        if (i >= 2 && (currentVertex.x !== vertices[0].x || currentVertex.y !== vertices[0].y)) {\n            fillElement.add(firstIndex, prevIndex, currentIndex);\n            elementGroup.elementLength++;\n        }\n\n        if (i >= 1) {\n            outlineElement.add(prevIndex, currentIndex);\n            elementGroup.secondElementLength++;\n        }\n\n        prevIndex = currentIndex;\n    }\n};\n\nFillBucket.prototype.hasData = function() {\n    return !!this.elementGroups.current;\n};\n","'use strict';\n\nvar ElementGroups = require('./elementgroups.js');\n\nmodule.exports = LineBucket;\n\nfunction LineBucket(info, buffers, placement, elementGroups) {\n    this.info = info;\n    this.buffers = buffers;\n    this.elementGroups = elementGroups || new ElementGroups(buffers.lineVertex, buffers.lineElement);\n}\n\nLineBucket.prototype.addFeatures = function() {\n    var features = this.features;\n    for (var i = 0; i < features.length; i++) {\n        var feature = features[i];\n        this.addFeature(feature.loadGeometry());\n    }\n};\n\nLineBucket.prototype.addFeature = function(lines) {\n    var info = this.info;\n    for (var i = 0; i < lines.length; i++) {\n        this.addLine(lines[i], info['line-join'], info['line-cap'],\n                info['line-miter-limit'], info['line-round-limit']);\n    }\n};\n\nLineBucket.prototype.addLine = function(vertices, join, cap, miterLimit, roundLimit) {\n    if (vertices.length < 2) {\n        //console.warn('a line must have at least two vertices');\n        return;\n    }\n\n    if (join === 'bevel') miterLimit = 1.05;\n\n    var len = vertices.length,\n        firstVertex = vertices[0],\n        lastVertex = vertices[len - 1],\n        closed = firstVertex.equals(lastVertex);\n\n    var lineVertex = this.buffers.lineVertex;\n    var lineElement = this.buffers.lineElement;\n\n    // we could be more precies, but it would only save a negligible amount of space\n    this.elementGroups.makeRoomFor(len * 4);\n    var elementGroup = this.elementGroups.current;\n    var vertexStartIndex = elementGroup.vertexStartIndex;\n\n    if (len == 2 && closed) {\n        // console.warn('a line may not have coincident points');\n        return;\n    }\n\n    var beginCap = cap,\n        endCap = closed ? 'butt' : cap,\n        flip = 1,\n        distance = 0,\n        currentVertex, prevVertex,  nextVertex, prevNormal,  nextNormal;\n\n    // the last three vertices added\n    var e1, e2, e3;\n\n    if (closed) {\n        currentVertex = vertices[len - 2];\n        nextNormal = firstVertex.sub(currentVertex)._unit()._perp();\n    }\n\n    for (var i = 0; i < len; i++) {\n\n        nextVertex = closed && i === len - 1 ?\n            vertices[1] : // if the line is closed, we treat the last vertex like the first\n            vertices[i + 1]; // just the next vertex\n\n        // if two consecutive vertices exist, skip the current one\n        if (nextVertex && vertices[i].equals(nextVertex)) continue;\n\n        if (nextNormal) prevNormal = nextNormal;\n        if (currentVertex) prevVertex = currentVertex;\n\n        currentVertex = vertices[i];\n\n        // Calculate how far along the line the currentVertex is\n        if (prevVertex) distance += currentVertex.dist(prevVertex);\n\n        // Calculate the normal towards the next vertex in this line. In case\n        // there is no next vertex, pretend that the line is continuing straight,\n        // meaning that we are just using the previous normal.\n        nextNormal = nextVertex ? nextVertex.sub(currentVertex)._unit()._perp() : prevNormal;\n\n        // If we still don't have a previous normal, this is the beginning of a\n        // non-closed line, so we're doing a straight \"join\".\n        prevNormal = prevNormal || nextNormal;\n\n        // Determine the normal of the join extrusion. It is the angle bisector\n        // of the segments between the previous line and the next line.\n        var joinNormal = prevNormal.add(nextNormal)._unit();\n\n        /*  joinNormal     prevNormal\n         *             ↖      ↑\n         *                .________. prevVertex\n         *                |\n         * nextNormal  ←  |  currentVertex\n         *                |\n         *     nextVertex !\n         *\n         */\n\n        // Calculate the length of the miter (the ratio of the miter to the width).\n        // Find the cosine of the angle between the next and join normals\n        // using dot product. The inverse of that is the miter length.\n        var cosHalfAngle = joinNormal.x * nextNormal.x + joinNormal.y * nextNormal.y;\n        var miterLength = 1 / cosHalfAngle;\n\n        // Whether any vertices have been\n        var startOfLine = e1 === undefined || e2 === undefined;\n\n        // The join if a middle vertex, otherwise the cap.\n        var middleVertex = prevVertex && nextVertex;\n        var currentJoin = middleVertex ? join : nextVertex ? beginCap : endCap;\n\n        if (middleVertex && currentJoin === 'round' && miterLength < roundLimit) {\n            currentJoin = 'miter';\n        }\n\n        if (currentJoin === 'miter' && miterLength > miterLimit) {\n            currentJoin = 'bevel';\n        }\n\n        if (currentJoin === 'bevel') {\n            // The maximum extrude length is 63 / 256 = 4 times the width of the line\n            // so if miterLength >= 4 we need to draw a different type of bevel where.\n            if (miterLength > 4) currentJoin = 'flipbevel';\n\n            // If the miterLength is really small and the line bevel wouldn't be visible,\n            // just draw a miter join to save a triangle.\n            if (miterLength < miterLimit) currentJoin = 'miter';\n        }\n\n        // Mitered joins\n        if (currentJoin === 'miter') {\n            // scale the unit vector by the miter length\n            joinNormal._mult(miterLength);\n            addCurrentVertex(joinNormal, 0, 0, false);\n\n        } else if (currentJoin === 'flipbevel') {\n            // miter is too big, flip the direction to make a beveled join\n\n            if (miterLength > 100) {\n                // Almost parallel lines\n                flip = -flip;\n                joinNormal = nextNormal;\n\n            } else {\n                var bevelLength = miterLength * prevNormal.add(nextNormal).mag() / prevNormal.sub(nextNormal).mag();\n                joinNormal._perp()._mult(flip * bevelLength);\n                flip = -flip;\n            }\n            addCurrentVertex(joinNormal, 0, 0, false);\n\n        // All other types of joins\n        } else {\n\n            var offsetA, offsetB;\n            if (currentJoin === 'bevel') {\n                var dir = prevNormal.x * nextNormal.y - prevNormal.y * nextNormal.x;\n                var offset = -Math.sqrt(miterLength * miterLength - 1);\n                if (flip * dir > 0) {\n                    offsetB = 0;\n                    offsetA = offset;\n                } else {\n                    offsetA = 0;\n                    offsetB = offset;\n                }\n            } else if (currentJoin === 'square') {\n                offsetA = offsetB = 1;\n            } else {\n                offsetA = offsetB = 0;\n            }\n\n            // Close previous segment with a butt or a square cap or bevel\n            if (!startOfLine) {\n                addCurrentVertex(prevNormal, offsetA, offsetB, false);\n            }\n\n            // Add round cap or linejoin at end of segment\n            if (!startOfLine && currentJoin === 'round') {\n                addCurrentVertex(prevNormal, 1, 1, true);\n            }\n\n            // Segment include cap are done, unset vertices to disconnect segments.\n            // Or leave them to create a bevel.\n            if (startOfLine || currentJoin !== 'bevel') {\n                e1 = e2 = -1;\n                flip = 1;\n            }\n\n            // Add round cap before first segment\n            if (startOfLine && beginCap === 'round') {\n                addCurrentVertex(nextNormal, -1, -1, true);\n            }\n\n            // Start next segment with a butt or square cap or bevel\n            if (nextVertex) {\n                addCurrentVertex(nextNormal, -offsetA, -offsetB, false);\n            }\n        }\n\n    }\n\n\n    /*\n     * Adds two vertices to the buffer that are\n     * normal and -normal from the currentVertex.\n     *\n     * endBox moves the extrude one unit in the direction of the line\n     * to create square or round cap.\n     *\n     * endLeft and endRight shifts the extrude along the line\n     * endLeft === 1 moves the extrude in the direction of the line\n     * endLeft === -1 moves the extrude in the reverse direction\n     */\n    function addCurrentVertex(normal, endLeft, endRight, round) {\n\n        var tx = round ? 1 : 0;\n        var extrude;\n\n        extrude = normal.mult(flip);\n        if (endLeft) extrude._sub(normal.perp()._mult(endLeft));\n        e3 = lineVertex.add(currentVertex, extrude, tx, 0, distance) - vertexStartIndex;\n        if (e1 >= 0 && e2 >= 0) {\n            lineElement.add(e1, e2, e3);\n            elementGroup.elementLength++;\n        }\n        e1 = e2;\n        e2 = e3;\n\n        extrude = normal.mult(-flip);\n        if (endRight) extrude._sub(normal.perp()._mult(endRight));\n        e3 = lineVertex.add(currentVertex, extrude, tx, 1, distance) - vertexStartIndex;\n        if (e1 >= 0 && e2 >= 0) {\n            lineElement.add(e1, e2, e3);\n            elementGroup.elementLength++;\n        }\n        e1 = e2;\n        e2 = e3;\n\n        elementGroup.vertexLength += 2;\n    }\n};\n\nLineBucket.prototype.hasData = function() {\n    return !!this.elementGroups.current;\n};\n","'use strict';\n\nmodule.exports = RasterBucket;\n\nfunction RasterBucket(info) {\n    this.info = info;\n}\n","'use strict';\n\nvar ElementGroups = require('./elementgroups.js');\nvar Anchor = require('../symbol/anchor.js');\nvar interpolate = require('../symbol/interpolate.js');\nvar Point = require('point-geometry');\nvar resolveTokens = require('../util/token.js');\nvar Placement = require('../symbol/placement.js');\nvar Shaping = require('../symbol/shaping.js');\nvar resolveText = require('../symbol/resolvetext.js');\n\nmodule.exports = SymbolBucket;\n\nvar fullRange = [2 * Math.PI , 0];\n\nfunction SymbolBucket(info, buffers, collision, elementGroups) {\n    this.info = info;\n    this.buffers = buffers;\n    this.collision = collision;\n\n    if (info['symbol-placement'] === 'line') {\n        if (!info.hasOwnProperty('text-rotation-alignment')) {\n            info['text-rotation-alignment'] = 'map';\n        }\n        if (!info.hasOwnProperty('icon-rotation-alignment')) {\n            info['icon-rotation-alignment'] = 'map';\n        }\n\n        info['symbol-avoid-edges'] = true;\n    }\n\n    if (elementGroups) {\n        this.elementGroups = elementGroups;\n    } else {\n        this.elementGroups = {\n            text: new ElementGroups(buffers.glyphVertex),\n            icon: new ElementGroups(buffers.iconVertex)\n        };\n    }\n}\n\nSymbolBucket.prototype.addFeatures = function() {\n    var info = this.info;\n    var features = this.features;\n    var textFeatures = this.textFeatures;\n\n    var horizontalAlign = 0.5;\n    if (info['text-horizontal-align'] === 'right') horizontalAlign = 1;\n    else if (info['text-horizontal-align'] === 'left') horizontalAlign = 0;\n\n    var verticalAlign = 0.5;\n    if (info['text-vertical-align'] === 'bottom') verticalAlign = 1;\n    else if (info['text-vertical-align'] === 'top') verticalAlign = 0;\n\n    var justify = 0.5;\n    if (info['text-justify'] === 'right') justify = 1;\n    else if (info['text-justify'] === 'left') justify = 0;\n\n    var oneEm = 24;\n    var lineHeight = info['text-line-height'] * oneEm;\n    var maxWidth = info['symbol-placement'] !== 'line' && info['text-max-width'] * oneEm;\n    var spacing = info['text-letter-spacing'] * oneEm;\n    var fontstack = info['text-font'];\n    var textOffset = [info['text-offset'][0] * oneEm, info['text-offset'][1] * oneEm];\n\n    for (var k = 0; k < features.length; k++) {\n\n        var feature = features[k];\n        var text = textFeatures[k];\n        var lines = feature.loadGeometry();\n\n        var shaping = false;\n        if (text) {\n            shaping = Shaping.shape(text, fontstack, this.stacks, maxWidth,\n                    lineHeight, horizontalAlign, verticalAlign, justify, spacing, textOffset);\n        }\n\n        var image = false;\n        if (this.sprite && this.info['icon-image']) {\n            image = this.sprite[resolveTokens(feature.properties, info['icon-image'])];\n\n            if (image) {\n                // match glyph tex object. TODO change\n                image.w = image.width;\n                image.h = image.height;\n\n                if (image.sdf) this.elementGroups.sdfIcons = true;\n            }\n        }\n\n        if (!shaping && !image) continue;\n        this.addFeature(lines, this.stacks, shaping, image);\n    }\n};\n\nfunction byScale(a, b) {\n    return a.scale - b.scale;\n}\n\nSymbolBucket.prototype.addFeature = function(lines, faces, shaping, image) {\n    var info = this.info;\n    var collision = this.collision;\n\n    var minScale = 0.5;\n    var glyphSize = 24;\n\n    var horizontalText = info['text-rotation-alignment'] === 'viewport',\n        horizontalIcon = info['icon-rotation-alignment'] === 'viewport',\n        fontScale = info['text-max-size'] / glyphSize,\n        textBoxScale = collision.tilePixelRatio * fontScale,\n        iconBoxScale = collision.tilePixelRatio * info['icon-max-size'],\n        iconWithoutText = info['text-optional'] || !shaping,\n        textWithoutIcon = info['icon-optional'] || !image,\n        avoidEdges = info['symbol-avoid-edges'];\n\n    for (var i = 0; i < lines.length; i++) {\n\n        var line = lines[i];\n        var anchors;\n\n        if (info['symbol-placement'] === 'line') {\n            // Line labels\n            anchors = interpolate(line, info['symbol-min-distance'], minScale, collision.maxPlacementScale, collision.tilePixelRatio);\n\n            // Sort anchors by segment so that we can start placement with the\n            // anchors that can be shown at the lowest zoom levels.\n            anchors.sort(byScale);\n\n        } else {\n            // Point labels\n            anchors = [new Anchor(line[0].x, line[0].y, 0, minScale)];\n        }\n\n\n        // TODO: figure out correct ascender height.\n        var origin = new Point(0, -17);\n\n        for (var j = 0, len = anchors.length; j < len; j++) {\n            var anchor = anchors[j];\n            var inside = !(anchor.x < 0 || anchor.x > 4096 || anchor.y < 0 || anchor.y > 4096);\n\n            if (avoidEdges && !inside) continue;\n\n            // Calculate the scales at which the text and icons can be first shown without overlap\n            var glyph;\n            var icon;\n            var glyphScale = null;\n            var iconScale = null;\n\n            if (shaping) {\n                glyph = Placement.getGlyphs(anchor, origin, shaping, faces, textBoxScale, horizontalText, line, info);\n                glyphScale = info['text-allow-overlap'] ? glyph.minScale\n                    : collision.getPlacementScale(glyph.boxes, glyph.minScale, avoidEdges);\n                if (!glyphScale && !iconWithoutText) continue;\n            }\n\n            if (image) {\n                icon = Placement.getIcon(anchor, image, iconBoxScale, line, info);\n                iconScale = info['icon-allow-overlap'] ? icon.minScale\n                    : collision.getPlacementScale(icon.boxes, icon.minScale, avoidEdges);\n                if (!iconScale && !textWithoutIcon) continue;\n            }\n\n            if (!iconWithoutText && !textWithoutIcon) {\n                iconScale = glyphScale = Math.max(iconScale, glyphScale);\n            } else if (!textWithoutIcon && glyphScale) {\n                glyphScale = Math.max(iconScale, glyphScale);\n            } else if (!iconWithoutText && iconScale) {\n                iconScale = Math.max(iconScale, glyphScale);\n            }\n\n            // Get the rotation ranges it is safe to show the glyphs\n            var glyphRange = (!glyphScale || info['text-allow-overlap']) ? fullRange\n                : collision.getPlacementRange(glyph.boxes, glyphScale, horizontalText);\n            var iconRange = (!iconScale || info['icon-allow-overlap']) ? fullRange\n                : collision.getPlacementRange(icon.boxes, iconScale, horizontalIcon);\n\n            var maxRange = [\n                Math.min(iconRange[0], glyphRange[0]),\n                Math.max(iconRange[1], glyphRange[1])];\n\n            if (!iconWithoutText && !textWithoutIcon) {\n                iconRange = glyphRange = maxRange;\n            } else if (!textWithoutIcon) {\n                glyphRange = maxRange;\n            } else if (!iconWithoutText) {\n                iconRange = maxRange;\n            }\n\n            // Insert final placement into collision tree and add glyphs/icons to buffers\n            if (glyphScale) {\n                if (!info['text-ignore-placement']) {\n                    collision.insert(glyph.boxes, anchor, glyphScale, glyphRange, horizontalText);\n                }\n                if (inside) this.addSymbols(this.buffers.glyphVertex, this.elementGroups.text, glyph.shapes, glyphScale, glyphRange);\n            }\n\n            if (iconScale) {\n                if (!info['icon-ignore-placement']) {\n                    collision.insert(icon.boxes, anchor, iconScale, iconRange, horizontalIcon);\n                }\n                if (inside) this.addSymbols(this.buffers.iconVertex, this.elementGroups.icon, icon.shapes, iconScale, iconRange);\n            }\n\n        }\n    }\n};\n\nSymbolBucket.prototype.addSymbols = function(buffer, elementGroups, symbols, scale, placementRange) {\n\n    var zoom = this.collision.zoom;\n\n    elementGroups.makeRoomFor(0);\n    var elementGroup = elementGroups.current;\n\n    var placementZoom = Math.log(scale) / Math.LN2 + zoom;\n\n    for (var k = 0; k < symbols.length; k++) {\n\n        var symbol = symbols[k],\n            tl = symbol.tl,\n            tr = symbol.tr,\n            bl = symbol.bl,\n            br = symbol.br,\n            tex = symbol.tex,\n            angle = symbol.angle,\n            anchor = symbol.anchor,\n\n\n            minZoom = Math.max(zoom + Math.log(symbol.minScale) / Math.LN2, placementZoom),\n            maxZoom = Math.min(zoom + Math.log(symbol.maxScale) / Math.LN2, 25);\n\n        if (maxZoom <= minZoom) continue;\n\n        // Lower min zoom so that while fading out the label it can be shown outside of collision-free zoom levels\n        if (minZoom === placementZoom) minZoom = 0;\n\n        // first triangle\n        buffer.add(anchor.x, anchor.y, tl.x, tl.y, tex.x, tex.y, angle, minZoom, placementRange, maxZoom, placementZoom);\n        buffer.add(anchor.x, anchor.y, tr.x, tr.y, tex.x + tex.w, tex.y, angle, minZoom, placementRange, maxZoom, placementZoom);\n        buffer.add(anchor.x, anchor.y, bl.x, bl.y, tex.x, tex.y + tex.h, angle, minZoom, placementRange, maxZoom, placementZoom);\n\n        // second triangle\n        buffer.add(anchor.x, anchor.y, tr.x, tr.y, tex.x + tex.w, tex.y, angle, minZoom, placementRange, maxZoom, placementZoom);\n        buffer.add(anchor.x, anchor.y, bl.x, bl.y, tex.x, tex.y + tex.h, angle, minZoom, placementRange, maxZoom, placementZoom);\n        buffer.add(anchor.x, anchor.y, br.x, br.y, tex.x + tex.w, tex.y + tex.h, angle, minZoom, placementRange, maxZoom, placementZoom);\n\n        elementGroup.vertexLength += 6;\n    }\n\n};\n\nSymbolBucket.prototype.getDependencies = function(tile, actor, callback) {\n    var firstdone = false;\n    var firsterr;\n    this.getTextDependencies(tile, actor, done);\n    this.getIconDependencies(tile, actor, done);\n    function done(err) {\n        if (err || firstdone) callback(err);\n        firstdone = true;\n        firsterr = err;\n    }\n};\n\nSymbolBucket.prototype.getIconDependencies = function(tile, actor, callback) {\n    var bucket = this;\n    if (this.info['icon-image']) {\n        if (SymbolBucket.sprite) {\n            this.sprite = SymbolBucket.sprite;\n            callback();\n        } else {\n            actor.send('get sprite json', {}, function(err, data) {\n                SymbolBucket.sprite = bucket.sprite = data.sprite;\n                callback(err);\n            });\n        }\n    } else {\n        callback();\n    }\n};\n\nSymbolBucket.prototype.getTextDependencies = function(tile, actor, callback) {\n    var features = this.features;\n    var info = this.info;\n\n    if (tile.stacks === undefined) tile.stacks = {};\n    var stacks = this.stacks = tile.stacks;\n    var fontstack = info['text-font'];\n    if (stacks[fontstack] === undefined) {\n        stacks[fontstack] = { glyphs: {}, rects: {} };\n    }\n    var stack = stacks[fontstack];\n\n    var data = resolveText(features, info, stack.glyphs);\n    this.textFeatures = data.textFeatures;\n\n    actor.send('get glyphs', {\n        id: tile.id,\n        fontstack: fontstack,\n        codepoints: data.codepoints\n    }, function(err, newstack) {\n        if (err) return callback(err);\n\n        var newglyphs = newstack.glyphs;\n        var newrects = newstack.rects;\n        var glyphs = stack.glyphs;\n        var rects = stack.rects;\n\n        for (var codepoint in newglyphs) {\n            glyphs[codepoint] = newglyphs[codepoint];\n            rects[codepoint] = newrects[codepoint];\n        }\n\n        callback();\n    });\n};\n\nSymbolBucket.prototype.hasData = function() {\n    return !!this.elementGroups.text.current || !!this.elementGroups.icon.current;\n};\n","'use strict';\n\nmodule.exports = LatLng;\n\nfunction LatLng(lat, lng) {\n    if (isNaN(lat) || isNaN(lng)) {\n        throw new Error('Invalid LatLng object: (' + lat + ', ' + lng + ')');\n    }\n    this.lat = +lat;\n    this.lng = +lng;\n}\n\n\n// constructs LatLng from an array if necessary\n\nLatLng.convert = function (a) {\n    if (a instanceof LatLng) {\n        return a;\n    }\n    if (Array.isArray(a)) {\n        return new LatLng(a[0], a[1]);\n    }\n    return a;\n};\n","'use strict';\n\nmodule.exports = LatLngBounds;\n\nvar LatLng = require('./latlng.js');\n\nfunction LatLngBounds(sw, ne) {\n    if (!sw) return;\n\n    var latlngs = ne ? [sw, ne] : sw;\n\n    for (var i = 0, len = latlngs.length; i < len; i++) {\n        this.extend(latlngs[i]);\n    }\n}\n\nLatLngBounds.prototype = {\n\n    // extend the bounds to contain the given point or bounds\n    extend: function (obj) {\n        var sw = this._sw,\n            ne = this._ne,\n            sw2, ne2;\n\n        if (obj instanceof LatLng) {\n            sw2 = obj;\n            ne2 = obj;\n\n        } else if (obj instanceof LatLngBounds) {\n            sw2 = obj._sw;\n            ne2 = obj._ne;\n\n            if (!sw2 || !ne2) return this;\n\n        } else {\n            return obj ? this.extend(LatLng.convert(obj) || LatLngBounds.convert(obj)) : this;\n        }\n\n        if (!sw && !ne) {\n            this._sw = new LatLng(sw2.lat, sw2.lng);\n            this._ne = new LatLng(ne2.lat, ne2.lng);\n\n        } else {\n            sw.lat = Math.min(sw2.lat, sw.lat);\n            sw.lng = Math.min(sw2.lng, sw.lng);\n            ne.lat = Math.max(ne2.lat, ne.lat);\n            ne.lng = Math.max(ne2.lng, ne.lng);\n        }\n\n        return this;\n    },\n\n    getCenter: function () {\n        return new LatLng((this._sw.lat + this._ne.lat) / 2, (this._sw.lng + this._ne.lng) / 2);\n    },\n\n    getSouthWest: function () { return this._sw; },\n    getNorthEast: function () { return this._ne; },\n    getNorthWest: function () { return new LatLng(this.getNorth(), this.getWest()); },\n    getSouthEast: function () { return new LatLng(this.getSouth(), this.getEast()); },\n\n    getWest:  function () { return this._sw.lng; },\n    getSouth: function () { return this._sw.lat; },\n    getEast:  function () { return this._ne.lng; },\n    getNorth: function () { return this._ne.lat; }\n};\n\n// constructs LatLngBounds from an array if necessary\nLatLngBounds.convert = function (a) {\n    if (!a || a instanceof LatLngBounds) return a;\n    return new LatLngBounds(a);\n};\n","'use strict';\n\nvar LatLng = require('./latlng.js'),\n    Point = require('point-geometry');\n\nmodule.exports = Transform;\n\n// A single transform, generally used for a single tile to be scaled, rotated, and zoomed.\n\nfunction Transform(minZoom, maxZoom) {\n    this.tileSize = 512; // constant\n\n    this._minZoom = minZoom || 0;\n    this._maxZoom = maxZoom || 22;\n\n    this.latRange = [-85, 85];\n\n    this.width = 0;\n    this.height = 0;\n    this.zoom = 0;\n    this.center = new LatLng(0, 0);\n    this.angle = 0;\n}\n\nTransform.prototype = {\n    get minZoom() { return this._minZoom; },\n    set minZoom(zoom) {\n        this._minZoom = zoom;\n        this.zoom = Math.max(this.zoom, zoom);\n    },\n\n    get maxZoom() { return this._maxZoom; },\n    set maxZoom(zoom) {\n        this._maxZoom = zoom;\n        this.zoom = Math.min(this.zoom, zoom);\n    },\n\n    get worldSize() {\n        return this.tileSize * this.scale;\n    },\n\n    get centerPoint() {\n        return this.size._div(2);\n    },\n\n    get size() {\n        return new Point(this.width, this.height);\n    },\n\n    get bearing() {\n        return -this.angle / Math.PI * 180;\n    },\n    set bearing(bearing) {\n        // confine the angle to within [-180,180]\n        bearing = ((((bearing + 180) % 360) + 360) % 360) - 180;\n        this.angle = -bearing * Math.PI / 180;\n    },\n\n    get zoom() { return this._zoom; },\n    set zoom(zoom) {\n        zoom = Math.min(Math.max(zoom, this.minZoom), this.maxZoom);\n        this._zoom = zoom;\n        this.scale = this.zoomScale(zoom);\n        this.tileZoom = Math.floor(zoom);\n        this.zoomFraction = zoom - this.tileZoom;\n        this._constrain();\n    },\n\n    zoomScale: function(zoom) { return Math.pow(2, zoom); },\n    scaleZoom: function(scale) { return Math.log(scale) / Math.LN2; },\n\n    project: function(latlng, worldSize) {\n        return new Point(\n            this.lngX(latlng.lng, worldSize),\n            this.latY(latlng.lat, worldSize));\n    },\n\n    unproject: function(point, worldSize) {\n        return new LatLng(\n            this.yLat(point.y, worldSize),\n            this.xLng(point.x, worldSize));\n    },\n\n    get x() { return this.lngX(this.center.lng); },\n    get y() { return this.latY(this.center.lat); },\n\n    get point() { return new Point(this.x, this.y); },\n\n    // lat/lon <-> absolute pixel coords convertion\n    lngX: function(lon, worldSize) {\n        return (180 + lon) * (worldSize || this.worldSize) / 360;\n    },\n    // latitude to absolute y coord\n    latY: function(lat, worldSize) {\n        var y = 180 / Math.PI * Math.log(Math.tan(Math.PI / 4 + lat * Math.PI / 360));\n        return (180 - y) * (worldSize || this.worldSize) / 360;\n    },\n\n    xLng: function(x, worldSize) {\n        return x * 360 / (worldSize || this.worldSize) - 180;\n    },\n    yLat: function(y, worldSize) {\n        var y2 = 180 - y * 360 / (worldSize || this.worldSize);\n        return 360 / Math.PI * Math.atan(Math.exp(y2 * Math.PI / 180)) - 90;\n    },\n\n    panBy: function(offset) {\n        var point = this.centerPoint._add(offset);\n        this.center = this.pointLocation(point);\n        this._constrain();\n    },\n\n    setZoomAround: function(zoom, center) {\n        var p = this.locationPoint(center),\n            p1 = this.size._sub(p),\n            latlng = this.pointLocation(p1);\n        this.zoom = zoom;\n        this.panBy(p1.sub(this.locationPoint(latlng)));\n    },\n\n    setBearingAround: function(bearing, center) {\n        var offset = this.locationPoint(center).sub(this.centerPoint);\n        this.panBy(offset);\n        this.bearing = bearing;\n        this.panBy(offset.mult(-1));\n    },\n\n    locationPoint: function(latlng) {\n        var p = this.project(latlng);\n        return this.centerPoint._sub(this.point._sub(p)._rotate(this.angle));\n    },\n\n    pointLocation: function(p) {\n        var p2 = this.centerPoint._sub(p)._rotate(-this.angle);\n        return this.unproject(this.point.sub(p2));\n    },\n\n    locationCoordinate: function(latlng) {\n        var k = this.zoomScale(this.tileZoom) / this.worldSize;\n        return {\n            column: this.lngX(latlng.lng) * k,\n            row: this.latY(latlng.lat) * k,\n            zoom: this.tileZoom\n        };\n    },\n\n    pointCoordinate: function(tileCenter, p) {\n        var zoomFactor = this.zoomScale(this.zoomFraction),\n            kt = this.zoomScale(this.tileZoom - tileCenter.zoom),\n            p2 = this.centerPoint._sub(p)._rotate(-this.angle)._div(this.tileSize * zoomFactor);\n\n        return {\n            column: tileCenter.column * kt - p2.x,\n            row: tileCenter.row * kt - p2.y,\n            zoom: this.tileZoom\n        };\n    },\n\n    _constrain: function() {\n        if (!this.center) return;\n\n        var minY, maxY, minX, maxX, sy, sx, x2, y2,\n            size = this.size;\n\n        if (this.latRange) {\n            minY = this.latY(this.latRange[1]);\n            maxY = this.latY(this.latRange[0]);\n            sy = maxY - minY < size.y ? size.y / (maxY - minY) : 0;\n        }\n\n        if (this.lngRange) {\n            minX = this.lngX(this.lngRange[0]);\n            maxX = this.lngX(this.lngRange[1]);\n            sx = maxX - minX < size.x ? size.x / (maxX - minX) : 0;\n        }\n\n        // how much the map should scale to fit the screen into given latitude/longitude ranges\n        var s = Math.max(sx || 0, sy || 0);\n\n        if (s) {\n            this.center = this.unproject(new Point(\n                sx ? (maxX + minX) / 2 : this.x,\n                sy ? (maxY + minY) / 2 : this.y));\n            this.zoom += this.scaleZoom(s);\n            return;\n        }\n\n        if (this.latRange) {\n            var y = this.y,\n                h2 = size.y / 2;\n\n            if (y - h2 < minY) y2 = minY + h2;\n            if (y + h2 > maxY) y2 = maxY - h2;\n        }\n\n        if (this.lngRange) {\n            var x = this.x,\n                w2 = size.x / 2;\n\n            if (x - w2 < minX) x2 = minX + w2;\n            if (x + w2 > maxX) x2 = maxX - w2;\n        }\n\n        // pan the map if the screen goes off the range\n        if (x2 !== undefined || y2 !== undefined) {\n            this.center = this.unproject(new Point(\n                x2 !== undefined ? x2 : this.x,\n                y2 !== undefined ? y2 : this.y));\n        }\n    }\n};\n","// Font data From Hershey Simplex Font\n// http://paulbourke.net/dataformats/hershey/\nvar simplex_font = {\n    \" \": [16, []],\n    \"!\": [10, [5, 21, 5, 7, -1, -1, 5, 2, 4, 1, 5, 0, 6, 1, 5, 2]],\n    \"\\\"\": [16, [4, 21, 4, 14, -1, -1, 12, 21, 12, 14]],\n    \"#\": [21, [11, 25, 4, -7, -1, -1, 17, 25, 10, -7, -1, -1, 4, 12, 18, 12, -1, -1, 3, 6, 17, 6]],\n    \"$\": [20, [8, 25, 8, -4, -1, -1, 12, 25, 12, -4, -1, -1, 17, 18, 15, 20, 12, 21, 8, 21, 5, 20, 3, 18, 3, 16, 4, 14, 5, 13, 7, 12, 13, 10, 15, 9, 16, 8, 17, 6, 17, 3, 15, 1, 12, 0, 8, 0, 5, 1, 3, 3]],\n    \"%\": [24, [21, 21, 3, 0, -1, -1, 8, 21, 10, 19, 10, 17, 9, 15, 7, 14, 5, 14, 3, 16, 3, 18, 4, 20, 6, 21, 8, 21, 10, 20, 13, 19, 16, 19, 19, 20, 21, 21, -1, -1, 17, 7, 15, 6, 14, 4, 14, 2, 16, 0, 18, 0, 20, 1, 21, 3, 21, 5, 19, 7, 17, 7]],\n    \"&\": [26, [23, 12, 23, 13, 22, 14, 21, 14, 20, 13, 19, 11, 17, 6, 15, 3, 13, 1, 11, 0, 7, 0, 5, 1, 4, 2, 3, 4, 3, 6, 4, 8, 5, 9, 12, 13, 13, 14, 14, 16, 14, 18, 13, 20, 11, 21, 9, 20, 8, 18, 8, 16, 9, 13, 11, 10, 16, 3, 18, 1, 20, 0, 22, 0, 23, 1, 23, 2]],\n    \"'\": [10, [5, 19, 4, 20, 5, 21, 6, 20, 6, 18, 5, 16, 4, 15]],\n    \"(\": [14, [11, 25, 9, 23, 7, 20, 5, 16, 4, 11, 4, 7, 5, 2, 7, -2, 9, -5, 11, -7]],\n    \")\": [14, [3, 25, 5, 23, 7, 20, 9, 16, 10, 11, 10, 7, 9, 2, 7, -2, 5, -5, 3, -7]],\n    \"*\": [16, [8, 21, 8, 9, -1, -1, 3, 18, 13, 12, -1, -1, 13, 18, 3, 12]],\n    \"+\": [26, [13, 18, 13, 0, -1, -1, 4, 9, 22, 9]],\n    \",\": [10, [6, 1, 5, 0, 4, 1, 5, 2, 6, 1, 6, -1, 5, -3, 4, -4]],\n    \"-\": [26, [4, 9, 22, 9]],\n    \".\": [10, [5, 2, 4, 1, 5, 0, 6, 1, 5, 2]],\n    \"/\": [22, [20, 25, 2, -7]],\n    \"0\": [20, [9, 21, 6, 20, 4, 17, 3, 12, 3, 9, 4, 4, 6, 1, 9, 0, 11, 0, 14, 1, 16, 4, 17, 9, 17, 12, 16, 17, 14, 20, 11, 21, 9, 21]],\n    \"1\": [20, [6, 17, 8, 18, 11, 21, 11, 0]],\n    \"2\": [20, [4, 16, 4, 17, 5, 19, 6, 20, 8, 21, 12, 21, 14, 20, 15, 19, 16, 17, 16, 15, 15, 13, 13, 10, 3, 0, 17, 0]],\n    \"3\": [20, [5, 21, 16, 21, 10, 13, 13, 13, 15, 12, 16, 11, 17, 8, 17, 6, 16, 3, 14, 1, 11, 0, 8, 0, 5, 1, 4, 2, 3, 4]],\n    \"4\": [20, [13, 21, 3, 7, 18, 7, -1, -1, 13, 21, 13, 0]],\n    \"5\": [20, [15, 21, 5, 21, 4, 12, 5, 13, 8, 14, 11, 14, 14, 13, 16, 11, 17, 8, 17, 6, 16, 3, 14, 1, 11, 0, 8, 0, 5, 1, 4, 2, 3, 4]],\n    \"6\": [20, [16, 18, 15, 20, 12, 21, 10, 21, 7, 20, 5, 17, 4, 12, 4, 7, 5, 3, 7, 1, 10, 0, 11, 0, 14, 1, 16, 3, 17, 6, 17, 7, 16, 10, 14, 12, 11, 13, 10, 13, 7, 12, 5, 10, 4, 7]],\n    \"7\": [20, [17, 21, 7, 0, -1, -1, 3, 21, 17, 21]],\n    \"8\": [20, [8, 21, 5, 20, 4, 18, 4, 16, 5, 14, 7, 13, 11, 12, 14, 11, 16, 9, 17, 7, 17, 4, 16, 2, 15, 1, 12, 0, 8, 0, 5, 1, 4, 2, 3, 4, 3, 7, 4, 9, 6, 11, 9, 12, 13, 13, 15, 14, 16, 16, 16, 18, 15, 20, 12, 21, 8, 21]],\n    \"9\": [20, [16, 14, 15, 11, 13, 9, 10, 8, 9, 8, 6, 9, 4, 11, 3, 14, 3, 15, 4, 18, 6, 20, 9, 21, 10, 21, 13, 20, 15, 18, 16, 14, 16, 9, 15, 4, 13, 1, 10, 0, 8, 0, 5, 1, 4, 3]],\n    \":\": [10, [5, 14, 4, 13, 5, 12, 6, 13, 5, 14, -1, -1, 5, 2, 4, 1, 5, 0, 6, 1, 5, 2]],\n    \";\": [10, [5, 14, 4, 13, 5, 12, 6, 13, 5, 14, -1, -1, 6, 1, 5, 0, 4, 1, 5, 2, 6, 1, 6, -1, 5, -3, 4, -4]],\n    \"<\": [24, [20, 18, 4, 9, 20, 0]],\n    \"=\": [26, [4, 12, 22, 12, -1, -1, 4, 6, 22, 6]],\n    \">\": [24, [4, 18, 20, 9, 4, 0]],\n    \"?\": [18, [3, 16, 3, 17, 4, 19, 5, 20, 7, 21, 11, 21, 13, 20, 14, 19, 15, 17, 15, 15, 14, 13, 13, 12, 9, 10, 9, 7, -1, -1, 9, 2, 8, 1, 9, 0, 10, 1, 9, 2]],\n    \"@\": [27, [18, 13, 17, 15, 15, 16, 12, 16, 10, 15, 9, 14, 8, 11, 8, 8, 9, 6, 11, 5, 14, 5, 16, 6, 17, 8, -1, -1, 12, 16, 10, 14, 9, 11, 9, 8, 10, 6, 11, 5, -1, -1, 18, 16, 17, 8, 17, 6, 19, 5, 21, 5, 23, 7, 24, 10, 24, 12, 23, 15, 22, 17, 20, 19, 18, 20, 15, 21, 12, 21, 9, 20, 7, 19, 5, 17, 4, 15, 3, 12, 3, 9, 4, 6, 5, 4, 7, 2, 9, 1, 12, 0, 15, 0, 18, 1, 20, 2, 21, 3, -1, -1, 19, 16, 18, 8, 18, 6, 19, 5]],\n    \"A\": [18, [9, 21, 1, 0, -1, -1, 9, 21, 17, 0, -1, -1, 4, 7, 14, 7]],\n    \"B\": [21, [4, 21, 4, 0, -1, -1, 4, 21, 13, 21, 16, 20, 17, 19, 18, 17, 18, 15, 17, 13, 16, 12, 13, 11, -1, -1, 4, 11, 13, 11, 16, 10, 17, 9, 18, 7, 18, 4, 17, 2, 16, 1, 13, 0, 4, 0]],\n    \"C\": [21, [18, 16, 17, 18, 15, 20, 13, 21, 9, 21, 7, 20, 5, 18, 4, 16, 3, 13, 3, 8, 4, 5, 5, 3, 7, 1, 9, 0, 13, 0, 15, 1, 17, 3, 18, 5]],\n    \"D\": [21, [4, 21, 4, 0, -1, -1, 4, 21, 11, 21, 14, 20, 16, 18, 17, 16, 18, 13, 18, 8, 17, 5, 16, 3, 14, 1, 11, 0, 4, 0]],\n    \"E\": [19, [4, 21, 4, 0, -1, -1, 4, 21, 17, 21, -1, -1, 4, 11, 12, 11, -1, -1, 4, 0, 17, 0]],\n    \"F\": [18, [4, 21, 4, 0, -1, -1, 4, 21, 17, 21, -1, -1, 4, 11, 12, 11]],\n    \"G\": [21, [18, 16, 17, 18, 15, 20, 13, 21, 9, 21, 7, 20, 5, 18, 4, 16, 3, 13, 3, 8, 4, 5, 5, 3, 7, 1, 9, 0, 13, 0, 15, 1, 17, 3, 18, 5, 18, 8, -1, -1, 13, 8, 18, 8]],\n    \"H\": [22, [4, 21, 4, 0, -1, -1, 18, 21, 18, 0, -1, -1, 4, 11, 18, 11]],\n    \"I\": [8, [4, 21, 4, 0]],\n    \"J\": [16, [12, 21, 12, 5, 11, 2, 10, 1, 8, 0, 6, 0, 4, 1, 3, 2, 2, 5, 2, 7]],\n    \"K\": [21, [4, 21, 4, 0, -1, -1, 18, 21, 4, 7, -1, -1, 9, 12, 18, 0]],\n    \"L\": [17, [4, 21, 4, 0, -1, -1, 4, 0, 16, 0]],\n    \"M\": [24, [4, 21, 4, 0, -1, -1, 4, 21, 12, 0, -1, -1, 20, 21, 12, 0, -1, -1, 20, 21, 20, 0]],\n    \"N\": [22, [4, 21, 4, 0, -1, -1, 4, 21, 18, 0, -1, -1, 18, 21, 18, 0]],\n    \"O\": [22, [9, 21, 7, 20, 5, 18, 4, 16, 3, 13, 3, 8, 4, 5, 5, 3, 7, 1, 9, 0, 13, 0, 15, 1, 17, 3, 18, 5, 19, 8, 19, 13, 18, 16, 17, 18, 15, 20, 13, 21, 9, 21]],\n    \"P\": [21, [4, 21, 4, 0, -1, -1, 4, 21, 13, 21, 16, 20, 17, 19, 18, 17, 18, 14, 17, 12, 16, 11, 13, 10, 4, 10]],\n    \"Q\": [22, [9, 21, 7, 20, 5, 18, 4, 16, 3, 13, 3, 8, 4, 5, 5, 3, 7, 1, 9, 0, 13, 0, 15, 1, 17, 3, 18, 5, 19, 8, 19, 13, 18, 16, 17, 18, 15, 20, 13, 21, 9, 21, -1, -1, 12, 4, 18, -2]],\n    \"R\": [21, [4, 21, 4, 0, -1, -1, 4, 21, 13, 21, 16, 20, 17, 19, 18, 17, 18, 15, 17, 13, 16, 12, 13, 11, 4, 11, -1, -1, 11, 11, 18, 0]],\n    \"S\": [20, [17, 18, 15, 20, 12, 21, 8, 21, 5, 20, 3, 18, 3, 16, 4, 14, 5, 13, 7, 12, 13, 10, 15, 9, 16, 8, 17, 6, 17, 3, 15, 1, 12, 0, 8, 0, 5, 1, 3, 3]],\n    \"T\": [16, [8, 21, 8, 0, -1, -1, 1, 21, 15, 21]],\n    \"U\": [22, [4, 21, 4, 6, 5, 3, 7, 1, 10, 0, 12, 0, 15, 1, 17, 3, 18, 6, 18, 21]],\n    \"V\": [18, [1, 21, 9, 0, -1, -1, 17, 21, 9, 0]],\n    \"W\": [24, [2, 21, 7, 0, -1, -1, 12, 21, 7, 0, -1, -1, 12, 21, 17, 0, -1, -1, 22, 21, 17, 0]],\n    \"X\": [20, [3, 21, 17, 0, -1, -1, 17, 21, 3, 0]],\n    \"Y\": [18, [1, 21, 9, 11, 9, 0, -1, -1, 17, 21, 9, 11]],\n    \"Z\": [20, [17, 21, 3, 0, -1, -1, 3, 21, 17, 21, -1, -1, 3, 0, 17, 0]],\n    \"[\": [14, [4, 25, 4, -7, -1, -1, 5, 25, 5, -7, -1, -1, 4, 25, 11, 25, -1, -1, 4, -7, 11, -7]],\n    \"\\\\\": [14, [0, 21, 14, -3]],\n    \"]\": [14, [9, 25, 9, -7, -1, -1, 10, 25, 10, -7, -1, -1, 3, 25, 10, 25, -1, -1, 3, -7, 10, -7]],\n    \"^\": [16, [6, 15, 8, 18, 10, 15, -1, -1, 3, 12, 8, 17, 13, 12, -1, -1, 8, 17, 8, 0]],\n    \"_\": [16, [0, -2, 16, -2]],\n    \"`\": [10, [6, 21, 5, 20, 4, 18, 4, 16, 5, 15, 6, 16, 5, 17]],\n    \"a\": [19, [15, 14, 15, 0, -1, -1, 15, 11, 13, 13, 11, 14, 8, 14, 6, 13, 4, 11, 3, 8, 3, 6, 4, 3, 6, 1, 8, 0, 11, 0, 13, 1, 15, 3]],\n    \"b\": [19, [4, 21, 4, 0, -1, -1, 4, 11, 6, 13, 8, 14, 11, 14, 13, 13, 15, 11, 16, 8, 16, 6, 15, 3, 13, 1, 11, 0, 8, 0, 6, 1, 4, 3]],\n    \"c\": [18, [15, 11, 13, 13, 11, 14, 8, 14, 6, 13, 4, 11, 3, 8, 3, 6, 4, 3, 6, 1, 8, 0, 11, 0, 13, 1, 15, 3]],\n    \"d\": [19, [15, 21, 15, 0, -1, -1, 15, 11, 13, 13, 11, 14, 8, 14, 6, 13, 4, 11, 3, 8, 3, 6, 4, 3, 6, 1, 8, 0, 11, 0, 13, 1, 15, 3]],\n    \"e\": [18, [3, 8, 15, 8, 15, 10, 14, 12, 13, 13, 11, 14, 8, 14, 6, 13, 4, 11, 3, 8, 3, 6, 4, 3, 6, 1, 8, 0, 11, 0, 13, 1, 15, 3]],\n    \"f\": [12, [10, 21, 8, 21, 6, 20, 5, 17, 5, 0, -1, -1, 2, 14, 9, 14]],\n    \"g\": [19, [15, 14, 15, -2, 14, -5, 13, -6, 11, -7, 8, -7, 6, -6, -1, -1, 15, 11, 13, 13, 11, 14, 8, 14, 6, 13, 4, 11, 3, 8, 3, 6, 4, 3, 6, 1, 8, 0, 11, 0, 13, 1, 15, 3]],\n    \"h\": [19, [4, 21, 4, 0, -1, -1, 4, 10, 7, 13, 9, 14, 12, 14, 14, 13, 15, 10, 15, 0]],\n    \"i\": [8, [3, 21, 4, 20, 5, 21, 4, 22, 3, 21, -1, -1, 4, 14, 4, 0]],\n    \"j\": [10, [5, 21, 6, 20, 7, 21, 6, 22, 5, 21, -1, -1, 6, 14, 6, -3, 5, -6, 3, -7, 1, -7]],\n    \"k\": [17, [4, 21, 4, 0, -1, -1, 14, 14, 4, 4, -1, -1, 8, 8, 15, 0]],\n    \"l\": [8, [4, 21, 4, 0]],\n    \"m\": [30, [4, 14, 4, 0, -1, -1, 4, 10, 7, 13, 9, 14, 12, 14, 14, 13, 15, 10, 15, 0, -1, -1, 15, 10, 18, 13, 20, 14, 23, 14, 25, 13, 26, 10, 26, 0]],\n    \"n\": [19, [4, 14, 4, 0, -1, -1, 4, 10, 7, 13, 9, 14, 12, 14, 14, 13, 15, 10, 15, 0]],\n    \"o\": [19, [8, 14, 6, 13, 4, 11, 3, 8, 3, 6, 4, 3, 6, 1, 8, 0, 11, 0, 13, 1, 15, 3, 16, 6, 16, 8, 15, 11, 13, 13, 11, 14, 8, 14]],\n    \"p\": [19, [4, 14, 4, -7, -1, -1, 4, 11, 6, 13, 8, 14, 11, 14, 13, 13, 15, 11, 16, 8, 16, 6, 15, 3, 13, 1, 11, 0, 8, 0, 6, 1, 4, 3]],\n    \"q\": [19, [15, 14, 15, -7, -1, -1, 15, 11, 13, 13, 11, 14, 8, 14, 6, 13, 4, 11, 3, 8, 3, 6, 4, 3, 6, 1, 8, 0, 11, 0, 13, 1, 15, 3]],\n    \"r\": [13, [4, 14, 4, 0, -1, -1, 4, 8, 5, 11, 7, 13, 9, 14, 12, 14]],\n    \"s\": [17, [14, 11, 13, 13, 10, 14, 7, 14, 4, 13, 3, 11, 4, 9, 6, 8, 11, 7, 13, 6, 14, 4, 14, 3, 13, 1, 10, 0, 7, 0, 4, 1, 3, 3]],\n    \"t\": [12, [5, 21, 5, 4, 6, 1, 8, 0, 10, 0, -1, -1, 2, 14, 9, 14]],\n    \"u\": [19, [4, 14, 4, 4, 5, 1, 7, 0, 10, 0, 12, 1, 15, 4, -1, -1, 15, 14, 15, 0]],\n    \"v\": [16, [2, 14, 8, 0, -1, -1, 14, 14, 8, 0]],\n    \"w\": [22, [3, 14, 7, 0, -1, -1, 11, 14, 7, 0, -1, -1, 11, 14, 15, 0, -1, -1, 19, 14, 15, 0]],\n    \"x\": [17, [3, 14, 14, 0, -1, -1, 14, 14, 3, 0]],\n    \"y\": [16, [2, 14, 8, 0, -1, -1, 14, 14, 8, 0, 6, -4, 4, -6, 2, -7, 1, -7]],\n    \"z\": [17, [14, 14, 3, 0, -1, -1, 3, 14, 14, 14, -1, -1, 3, 0, 14, 0]],\n    \"{\": [14, [9, 25, 7, 24, 6, 23, 5, 21, 5, 19, 6, 17, 7, 16, 8, 14, 8, 12, 6, 10, -1, -1, 7, 24, 6, 22, 6, 20, 7, 18, 8, 17, 9, 15, 9, 13, 8, 11, 4, 9, 8, 7, 9, 5, 9, 3, 8, 1, 7, 0, 6, -2, 6, -4, 7, -6, -1, -1, 6, 8, 8, 6, 8, 4, 7, 2, 6, 1, 5, -1, 5, -3, 6, -5, 7, -6, 9, -7]],\n    \"|\": [8, [4, 25, 4, -7]],\n    \"}\": [14, [5, 25, 7, 24, 8, 23, 9, 21, 9, 19, 8, 17, 7, 16, 6, 14, 6, 12, 8, 10, -1, -1, 7, 24, 8, 22, 8, 20, 7, 18, 6, 17, 5, 15, 5, 13, 6, 11, 10, 9, 6, 7, 5, 5, 5, 3, 6, 1, 7, 0, 8, -2, 8, -4, 7, -6, -1, -1, 8, 8, 6, 6, 6, 4, 7, 2, 8, 1, 9, -1, 9, -3, 8, -5, 7, -6, 5, -7]],\n    \"~\": [24, [3, 6, 3, 8, 4, 11, 6, 12, 8, 12, 10, 11, 14, 8, 16, 7, 18, 7, 20, 8, 21, 10, -1, -1, 3, 8, 4, 10, 6, 11, 8, 11, 10, 10, 14, 7, 16, 6, 18, 6, 20, 7, 21, 10, 21, 12]],\n};\n\nmodule.exports = function textVertices(text, left, baseline, scale) {\n    scale = scale || 1;\n\n    var strokes = [],\n        i, len, j, len2, glyph, data, x, y, prev;\n\n    for (i = 0, len = text.length; i < len; i++) {\n        glyph = simplex_font[text[i]];\n        if (!glyph) continue;\n        prev = null;\n\n        for (j = 0, len2 = glyph[1].length; j < len2; j += 2) {\n            if (glyph[1][j] === -1 && glyph[1][j + 1] === -1) {\n                prev = null;\n\n            } else {\n                x = left + glyph[1][j] * scale;\n                y = baseline - glyph[1][j + 1] * scale;\n                if (prev) {\n                    strokes.push(prev.x, prev.y, x, y);\n                }\n                prev = {x: x, y: y};\n            }\n        }\n        left += glyph[0] * scale;\n    }\n\n    return strokes;\n};\n","/**\n * @fileoverview gl-matrix - High performance matrix and vector operations\n * @author Brandon Jones\n * @author Colin MacKenzie IV\n * @version 2.2.0\n */\n/* Copyright (c) 2013, Brandon Jones, Colin MacKenzie IV. All rights reserved.\n\nRedistribution and use in source and binary forms, with or without modification,\nare permitted provided that the following conditions are met:\n\n  * Redistributions of source code must retain the above copyright notice, this\n    list of conditions and the following disclaimer.\n  * Redistributions in binary form must reproduce the above copyright notice,\n    this list of conditions and the following disclaimer in the documentation\n    and/or other materials provided with the distribution.\n\nTHIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS \"AS IS\" AND\nANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED\nWARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE\nDISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR\nANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES\n(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;\nLOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON\nANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT\n(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS\nSOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */\n(function(e){\"use strict\";var t={};typeof exports==\"undefined\"?typeof define==\"function\"&&typeof define.amd==\"object\"&&define.amd?(t.exports={},define(function(){return t.exports})):t.exports=typeof window!=\"undefined\"?window:e:t.exports=exports,function(e){if(!t)var t=1e-6;if(!n)var n=typeof Float32Array!=\"undefined\"?Float32Array:Array;if(!r)var r=Math.random;var i={};i.setMatrixArrayType=function(e){n=e},typeof e!=\"undefined\"&&(e.glMatrix=i);var s={};s.create=function(){var e=new n(2);return e[0]=0,e[1]=0,e},s.clone=function(e){var t=new n(2);return t[0]=e[0],t[1]=e[1],t},s.fromValues=function(e,t){var r=new n(2);return r[0]=e,r[1]=t,r},s.copy=function(e,t){return e[0]=t[0],e[1]=t[1],e},s.set=function(e,t,n){return e[0]=t,e[1]=n,e},s.add=function(e,t,n){return e[0]=t[0]+n[0],e[1]=t[1]+n[1],e},s.subtract=function(e,t,n){return e[0]=t[0]-n[0],e[1]=t[1]-n[1],e},s.sub=s.subtract,s.multiply=function(e,t,n){return e[0]=t[0]*n[0],e[1]=t[1]*n[1],e},s.mul=s.multiply,s.divide=function(e,t,n){return e[0]=t[0]/n[0],e[1]=t[1]/n[1],e},s.div=s.divide,s.min=function(e,t,n){return e[0]=Math.min(t[0],n[0]),e[1]=Math.min(t[1],n[1]),e},s.max=function(e,t,n){return e[0]=Math.max(t[0],n[0]),e[1]=Math.max(t[1],n[1]),e},s.scale=function(e,t,n){return e[0]=t[0]*n,e[1]=t[1]*n,e},s.scaleAndAdd=function(e,t,n,r){return e[0]=t[0]+n[0]*r,e[1]=t[1]+n[1]*r,e},s.distance=function(e,t){var n=t[0]-e[0],r=t[1]-e[1];return Math.sqrt(n*n+r*r)},s.dist=s.distance,s.squaredDistance=function(e,t){var n=t[0]-e[0],r=t[1]-e[1];return n*n+r*r},s.sqrDist=s.squaredDistance,s.length=function(e){var t=e[0],n=e[1];return Math.sqrt(t*t+n*n)},s.len=s.length,s.squaredLength=function(e){var t=e[0],n=e[1];return t*t+n*n},s.sqrLen=s.squaredLength,s.negate=function(e,t){return e[0]=-t[0],e[1]=-t[1],e},s.normalize=function(e,t){var n=t[0],r=t[1],i=n*n+r*r;return i>0&&(i=1/Math.sqrt(i),e[0]=t[0]*i,e[1]=t[1]*i),e},s.dot=function(e,t){return e[0]*t[0]+e[1]*t[1]},s.cross=function(e,t,n){var r=t[0]*n[1]-t[1]*n[0];return e[0]=e[1]=0,e[2]=r,e},s.lerp=function(e,t,n,r){var i=t[0],s=t[1];return e[0]=i+r*(n[0]-i),e[1]=s+r*(n[1]-s),e},s.random=function(e,t){t=t||1;var n=r()*2*Math.PI;return e[0]=Math.cos(n)*t,e[1]=Math.sin(n)*t,e},s.transformMat2=function(e,t,n){var r=t[0],i=t[1];return e[0]=n[0]*r+n[2]*i,e[1]=n[1]*r+n[3]*i,e},s.transformMat2d=function(e,t,n){var r=t[0],i=t[1];return e[0]=n[0]*r+n[2]*i+n[4],e[1]=n[1]*r+n[3]*i+n[5],e},s.transformMat3=function(e,t,n){var r=t[0],i=t[1];return e[0]=n[0]*r+n[3]*i+n[6],e[1]=n[1]*r+n[4]*i+n[7],e},s.transformMat4=function(e,t,n){var r=t[0],i=t[1];return e[0]=n[0]*r+n[4]*i+n[12],e[1]=n[1]*r+n[5]*i+n[13],e},s.forEach=function(){var e=s.create();return function(t,n,r,i,s,o){var u,a;n||(n=2),r||(r=0),i?a=Math.min(i*n+r,t.length):a=t.length;for(u=r;u<a;u+=n)e[0]=t[u],e[1]=t[u+1],s(e,e,o),t[u]=e[0],t[u+1]=e[1];return t}}(),s.str=function(e){return\"vec2(\"+e[0]+\", \"+e[1]+\")\"},typeof e!=\"undefined\"&&(e.vec2=s);var o={};o.create=function(){var e=new n(3);return e[0]=0,e[1]=0,e[2]=0,e},o.clone=function(e){var t=new n(3);return t[0]=e[0],t[1]=e[1],t[2]=e[2],t},o.fromValues=function(e,t,r){var i=new n(3);return i[0]=e,i[1]=t,i[2]=r,i},o.copy=function(e,t){return e[0]=t[0],e[1]=t[1],e[2]=t[2],e},o.set=function(e,t,n,r){return e[0]=t,e[1]=n,e[2]=r,e},o.add=function(e,t,n){return e[0]=t[0]+n[0],e[1]=t[1]+n[1],e[2]=t[2]+n[2],e},o.subtract=function(e,t,n){return e[0]=t[0]-n[0],e[1]=t[1]-n[1],e[2]=t[2]-n[2],e},o.sub=o.subtract,o.multiply=function(e,t,n){return e[0]=t[0]*n[0],e[1]=t[1]*n[1],e[2]=t[2]*n[2],e},o.mul=o.multiply,o.divide=function(e,t,n){return e[0]=t[0]/n[0],e[1]=t[1]/n[1],e[2]=t[2]/n[2],e},o.div=o.divide,o.min=function(e,t,n){return e[0]=Math.min(t[0],n[0]),e[1]=Math.min(t[1],n[1]),e[2]=Math.min(t[2],n[2]),e},o.max=function(e,t,n){return e[0]=Math.max(t[0],n[0]),e[1]=Math.max(t[1],n[1]),e[2]=Math.max(t[2],n[2]),e},o.scale=function(e,t,n){return e[0]=t[0]*n,e[1]=t[1]*n,e[2]=t[2]*n,e},o.scaleAndAdd=function(e,t,n,r){return e[0]=t[0]+n[0]*r,e[1]=t[1]+n[1]*r,e[2]=t[2]+n[2]*r,e},o.distance=function(e,t){var n=t[0]-e[0],r=t[1]-e[1],i=t[2]-e[2];return Math.sqrt(n*n+r*r+i*i)},o.dist=o.distance,o.squaredDistance=function(e,t){var n=t[0]-e[0],r=t[1]-e[1],i=t[2]-e[2];return n*n+r*r+i*i},o.sqrDist=o.squaredDistance,o.length=function(e){var t=e[0],n=e[1],r=e[2];return Math.sqrt(t*t+n*n+r*r)},o.len=o.length,o.squaredLength=function(e){var t=e[0],n=e[1],r=e[2];return t*t+n*n+r*r},o.sqrLen=o.squaredLength,o.negate=function(e,t){return e[0]=-t[0],e[1]=-t[1],e[2]=-t[2],e},o.normalize=function(e,t){var n=t[0],r=t[1],i=t[2],s=n*n+r*r+i*i;return s>0&&(s=1/Math.sqrt(s),e[0]=t[0]*s,e[1]=t[1]*s,e[2]=t[2]*s),e},o.dot=function(e,t){return e[0]*t[0]+e[1]*t[1]+e[2]*t[2]},o.cross=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=n[0],u=n[1],a=n[2];return e[0]=i*a-s*u,e[1]=s*o-r*a,e[2]=r*u-i*o,e},o.lerp=function(e,t,n,r){var i=t[0],s=t[1],o=t[2];return e[0]=i+r*(n[0]-i),e[1]=s+r*(n[1]-s),e[2]=o+r*(n[2]-o),e},o.random=function(e,t){t=t||1;var n=r()*2*Math.PI,i=r()*2-1,s=Math.sqrt(1-i*i)*t;return e[0]=Math.cos(n)*s,e[1]=Math.sin(n)*s,e[2]=i*t,e},o.transformMat4=function(e,t,n){var r=t[0],i=t[1],s=t[2];return e[0]=n[0]*r+n[4]*i+n[8]*s+n[12],e[1]=n[1]*r+n[5]*i+n[9]*s+n[13],e[2]=n[2]*r+n[6]*i+n[10]*s+n[14],e},o.transformMat3=function(e,t,n){var r=t[0],i=t[1],s=t[2];return e[0]=r*n[0]+i*n[3]+s*n[6],e[1]=r*n[1]+i*n[4]+s*n[7],e[2]=r*n[2]+i*n[5]+s*n[8],e},o.transformQuat=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=n[0],u=n[1],a=n[2],f=n[3],l=f*r+u*s-a*i,c=f*i+a*r-o*s,h=f*s+o*i-u*r,p=-o*r-u*i-a*s;return e[0]=l*f+p*-o+c*-a-h*-u,e[1]=c*f+p*-u+h*-o-l*-a,e[2]=h*f+p*-a+l*-u-c*-o,e},o.forEach=function(){var e=o.create();return function(t,n,r,i,s,o){var u,a;n||(n=3),r||(r=0),i?a=Math.min(i*n+r,t.length):a=t.length;for(u=r;u<a;u+=n)e[0]=t[u],e[1]=t[u+1],e[2]=t[u+2],s(e,e,o),t[u]=e[0],t[u+1]=e[1],t[u+2]=e[2];return t}}(),o.str=function(e){return\"vec3(\"+e[0]+\", \"+e[1]+\", \"+e[2]+\")\"},typeof e!=\"undefined\"&&(e.vec3=o);var u={};u.create=function(){var e=new n(4);return e[0]=0,e[1]=0,e[2]=0,e[3]=0,e},u.clone=function(e){var t=new n(4);return t[0]=e[0],t[1]=e[1],t[2]=e[2],t[3]=e[3],t},u.fromValues=function(e,t,r,i){var s=new n(4);return s[0]=e,s[1]=t,s[2]=r,s[3]=i,s},u.copy=function(e,t){return e[0]=t[0],e[1]=t[1],e[2]=t[2],e[3]=t[3],e},u.set=function(e,t,n,r,i){return e[0]=t,e[1]=n,e[2]=r,e[3]=i,e},u.add=function(e,t,n){return e[0]=t[0]+n[0],e[1]=t[1]+n[1],e[2]=t[2]+n[2],e[3]=t[3]+n[3],e},u.subtract=function(e,t,n){return e[0]=t[0]-n[0],e[1]=t[1]-n[1],e[2]=t[2]-n[2],e[3]=t[3]-n[3],e},u.sub=u.subtract,u.multiply=function(e,t,n){return e[0]=t[0]*n[0],e[1]=t[1]*n[1],e[2]=t[2]*n[2],e[3]=t[3]*n[3],e},u.mul=u.multiply,u.divide=function(e,t,n){return e[0]=t[0]/n[0],e[1]=t[1]/n[1],e[2]=t[2]/n[2],e[3]=t[3]/n[3],e},u.div=u.divide,u.min=function(e,t,n){return e[0]=Math.min(t[0],n[0]),e[1]=Math.min(t[1],n[1]),e[2]=Math.min(t[2],n[2]),e[3]=Math.min(t[3],n[3]),e},u.max=function(e,t,n){return e[0]=Math.max(t[0],n[0]),e[1]=Math.max(t[1],n[1]),e[2]=Math.max(t[2],n[2]),e[3]=Math.max(t[3],n[3]),e},u.scale=function(e,t,n){return e[0]=t[0]*n,e[1]=t[1]*n,e[2]=t[2]*n,e[3]=t[3]*n,e},u.scaleAndAdd=function(e,t,n,r){return e[0]=t[0]+n[0]*r,e[1]=t[1]+n[1]*r,e[2]=t[2]+n[2]*r,e[3]=t[3]+n[3]*r,e},u.distance=function(e,t){var n=t[0]-e[0],r=t[1]-e[1],i=t[2]-e[2],s=t[3]-e[3];return Math.sqrt(n*n+r*r+i*i+s*s)},u.dist=u.distance,u.squaredDistance=function(e,t){var n=t[0]-e[0],r=t[1]-e[1],i=t[2]-e[2],s=t[3]-e[3];return n*n+r*r+i*i+s*s},u.sqrDist=u.squaredDistance,u.length=function(e){var t=e[0],n=e[1],r=e[2],i=e[3];return Math.sqrt(t*t+n*n+r*r+i*i)},u.len=u.length,u.squaredLength=function(e){var t=e[0],n=e[1],r=e[2],i=e[3];return t*t+n*n+r*r+i*i},u.sqrLen=u.squaredLength,u.negate=function(e,t){return e[0]=-t[0],e[1]=-t[1],e[2]=-t[2],e[3]=-t[3],e},u.normalize=function(e,t){var n=t[0],r=t[1],i=t[2],s=t[3],o=n*n+r*r+i*i+s*s;return o>0&&(o=1/Math.sqrt(o),e[0]=t[0]*o,e[1]=t[1]*o,e[2]=t[2]*o,e[3]=t[3]*o),e},u.dot=function(e,t){return e[0]*t[0]+e[1]*t[1]+e[2]*t[2]+e[3]*t[3]},u.lerp=function(e,t,n,r){var i=t[0],s=t[1],o=t[2],u=t[3];return e[0]=i+r*(n[0]-i),e[1]=s+r*(n[1]-s),e[2]=o+r*(n[2]-o),e[3]=u+r*(n[3]-u),e},u.random=function(e,t){return t=t||1,e[0]=r(),e[1]=r(),e[2]=r(),e[3]=r(),u.normalize(e,e),u.scale(e,e,t),e},u.transformMat4=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=t[3];return e[0]=n[0]*r+n[4]*i+n[8]*s+n[12]*o,e[1]=n[1]*r+n[5]*i+n[9]*s+n[13]*o,e[2]=n[2]*r+n[6]*i+n[10]*s+n[14]*o,e[3]=n[3]*r+n[7]*i+n[11]*s+n[15]*o,e},u.transformQuat=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=n[0],u=n[1],a=n[2],f=n[3],l=f*r+u*s-a*i,c=f*i+a*r-o*s,h=f*s+o*i-u*r,p=-o*r-u*i-a*s;return e[0]=l*f+p*-o+c*-a-h*-u,e[1]=c*f+p*-u+h*-o-l*-a,e[2]=h*f+p*-a+l*-u-c*-o,e},u.forEach=function(){var e=u.create();return function(t,n,r,i,s,o){var u,a;n||(n=4),r||(r=0),i?a=Math.min(i*n+r,t.length):a=t.length;for(u=r;u<a;u+=n)e[0]=t[u],e[1]=t[u+1],e[2]=t[u+2],e[3]=t[u+3],s(e,e,o),t[u]=e[0],t[u+1]=e[1],t[u+2]=e[2],t[u+3]=e[3];return t}}(),u.str=function(e){return\"vec4(\"+e[0]+\", \"+e[1]+\", \"+e[2]+\", \"+e[3]+\")\"},typeof e!=\"undefined\"&&(e.vec4=u);var a={};a.create=function(){var e=new n(4);return e[0]=1,e[1]=0,e[2]=0,e[3]=1,e},a.clone=function(e){var t=new n(4);return t[0]=e[0],t[1]=e[1],t[2]=e[2],t[3]=e[3],t},a.copy=function(e,t){return e[0]=t[0],e[1]=t[1],e[2]=t[2],e[3]=t[3],e},a.identity=function(e){return e[0]=1,e[1]=0,e[2]=0,e[3]=1,e},a.transpose=function(e,t){if(e===t){var n=t[1];e[1]=t[2],e[2]=n}else e[0]=t[0],e[1]=t[2],e[2]=t[1],e[3]=t[3];return e},a.invert=function(e,t){var n=t[0],r=t[1],i=t[2],s=t[3],o=n*s-i*r;return o?(o=1/o,e[0]=s*o,e[1]=-r*o,e[2]=-i*o,e[3]=n*o,e):null},a.adjoint=function(e,t){var n=t[0];return e[0]=t[3],e[1]=-t[1],e[2]=-t[2],e[3]=n,e},a.determinant=function(e){return e[0]*e[3]-e[2]*e[1]},a.multiply=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=t[3],u=n[0],a=n[1],f=n[2],l=n[3];return e[0]=r*u+i*f,e[1]=r*a+i*l,e[2]=s*u+o*f,e[3]=s*a+o*l,e},a.mul=a.multiply,a.rotate=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=t[3],u=Math.sin(n),a=Math.cos(n);return e[0]=r*a+i*u,e[1]=r*-u+i*a,e[2]=s*a+o*u,e[3]=s*-u+o*a,e},a.scale=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=t[3],u=n[0],a=n[1];return e[0]=r*u,e[1]=i*a,e[2]=s*u,e[3]=o*a,e},a.str=function(e){return\"mat2(\"+e[0]+\", \"+e[1]+\", \"+e[2]+\", \"+e[3]+\")\"},typeof e!=\"undefined\"&&(e.mat2=a);var f={};f.create=function(){var e=new n(6);return e[0]=1,e[1]=0,e[2]=0,e[3]=1,e[4]=0,e[5]=0,e},f.clone=function(e){var t=new n(6);return t[0]=e[0],t[1]=e[1],t[2]=e[2],t[3]=e[3],t[4]=e[4],t[5]=e[5],t},f.copy=function(e,t){return e[0]=t[0],e[1]=t[1],e[2]=t[2],e[3]=t[3],e[4]=t[4],e[5]=t[5],e},f.identity=function(e){return e[0]=1,e[1]=0,e[2]=0,e[3]=1,e[4]=0,e[5]=0,e},f.invert=function(e,t){var n=t[0],r=t[1],i=t[2],s=t[3],o=t[4],u=t[5],a=n*s-r*i;return a?(a=1/a,e[0]=s*a,e[1]=-r*a,e[2]=-i*a,e[3]=n*a,e[4]=(i*u-s*o)*a,e[5]=(r*o-n*u)*a,e):null},f.determinant=function(e){return e[0]*e[3]-e[1]*e[2]},f.multiply=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=t[3],u=t[4],a=t[5],f=n[0],l=n[1],c=n[2],h=n[3],p=n[4],d=n[5];return e[0]=r*f+i*c,e[1]=r*l+i*h,e[2]=s*f+o*c,e[3]=s*l+o*h,e[4]=f*u+c*a+p,e[5]=l*u+h*a+d,e},f.mul=f.multiply,f.rotate=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=t[3],u=t[4],a=t[5],f=Math.sin(n),l=Math.cos(n);return e[0]=r*l+i*f,e[1]=-r*f+i*l,e[2]=s*l+o*f,e[3]=-s*f+l*o,e[4]=l*u+f*a,e[5]=l*a-f*u,e},f.scale=function(e,t,n){var r=n[0],i=n[1];return e[0]=t[0]*r,e[1]=t[1]*i,e[2]=t[2]*r,e[3]=t[3]*i,e[4]=t[4]*r,e[5]=t[5]*i,e},f.translate=function(e,t,n){return e[0]=t[0],e[1]=t[1],e[2]=t[2],e[3]=t[3],e[4]=t[4]+n[0],e[5]=t[5]+n[1],e},f.str=function(e){return\"mat2d(\"+e[0]+\", \"+e[1]+\", \"+e[2]+\", \"+e[3]+\", \"+e[4]+\", \"+e[5]+\")\"},typeof e!=\"undefined\"&&(e.mat2d=f);var l={};l.create=function(){var e=new n(9);return e[0]=1,e[1]=0,e[2]=0,e[3]=0,e[4]=1,e[5]=0,e[6]=0,e[7]=0,e[8]=1,e},l.fromMat4=function(e,t){return e[0]=t[0],e[1]=t[1],e[2]=t[2],e[3]=t[4],e[4]=t[5],e[5]=t[6],e[6]=t[8],e[7]=t[9],e[8]=t[10],e},l.clone=function(e){var t=new n(9);return t[0]=e[0],t[1]=e[1],t[2]=e[2],t[3]=e[3],t[4]=e[4],t[5]=e[5],t[6]=e[6],t[7]=e[7],t[8]=e[8],t},l.copy=function(e,t){return e[0]=t[0],e[1]=t[1],e[2]=t[2],e[3]=t[3],e[4]=t[4],e[5]=t[5],e[6]=t[6],e[7]=t[7],e[8]=t[8],e},l.identity=function(e){return e[0]=1,e[1]=0,e[2]=0,e[3]=0,e[4]=1,e[5]=0,e[6]=0,e[7]=0,e[8]=1,e},l.transpose=function(e,t){if(e===t){var n=t[1],r=t[2],i=t[5];e[1]=t[3],e[2]=t[6],e[3]=n,e[5]=t[7],e[6]=r,e[7]=i}else e[0]=t[0],e[1]=t[3],e[2]=t[6],e[3]=t[1],e[4]=t[4],e[5]=t[7],e[6]=t[2],e[7]=t[5],e[8]=t[8];return e},l.invert=function(e,t){var n=t[0],r=t[1],i=t[2],s=t[3],o=t[4],u=t[5],a=t[6],f=t[7],l=t[8],c=l*o-u*f,h=-l*s+u*a,p=f*s-o*a,d=n*c+r*h+i*p;return d?(d=1/d,e[0]=c*d,e[1]=(-l*r+i*f)*d,e[2]=(u*r-i*o)*d,e[3]=h*d,e[4]=(l*n-i*a)*d,e[5]=(-u*n+i*s)*d,e[6]=p*d,e[7]=(-f*n+r*a)*d,e[8]=(o*n-r*s)*d,e):null},l.adjoint=function(e,t){var n=t[0],r=t[1],i=t[2],s=t[3],o=t[4],u=t[5],a=t[6],f=t[7],l=t[8];return e[0]=o*l-u*f,e[1]=i*f-r*l,e[2]=r*u-i*o,e[3]=u*a-s*l,e[4]=n*l-i*a,e[5]=i*s-n*u,e[6]=s*f-o*a,e[7]=r*a-n*f,e[8]=n*o-r*s,e},l.determinant=function(e){var t=e[0],n=e[1],r=e[2],i=e[3],s=e[4],o=e[5],u=e[6],a=e[7],f=e[8];return t*(f*s-o*a)+n*(-f*i+o*u)+r*(a*i-s*u)},l.multiply=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=t[3],u=t[4],a=t[5],f=t[6],l=t[7],c=t[8],h=n[0],p=n[1],d=n[2],v=n[3],m=n[4],g=n[5],y=n[6],b=n[7],w=n[8];return e[0]=h*r+p*o+d*f,e[1]=h*i+p*u+d*l,e[2]=h*s+p*a+d*c,e[3]=v*r+m*o+g*f,e[4]=v*i+m*u+g*l,e[5]=v*s+m*a+g*c,e[6]=y*r+b*o+w*f,e[7]=y*i+b*u+w*l,e[8]=y*s+b*a+w*c,e},l.mul=l.multiply,l.translate=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=t[3],u=t[4],a=t[5],f=t[6],l=t[7],c=t[8],h=n[0],p=n[1];return e[0]=r,e[1]=i,e[2]=s,e[3]=o,e[4]=u,e[5]=a,e[6]=h*r+p*o+f,e[7]=h*i+p*u+l,e[8]=h*s+p*a+c,e},l.rotate=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=t[3],u=t[4],a=t[5],f=t[6],l=t[7],c=t[8],h=Math.sin(n),p=Math.cos(n);return e[0]=p*r+h*o,e[1]=p*i+h*u,e[2]=p*s+h*a,e[3]=p*o-h*r,e[4]=p*u-h*i,e[5]=p*a-h*s,e[6]=f,e[7]=l,e[8]=c,e},l.scale=function(e,t,n){var r=n[0],i=n[1];return e[0]=r*t[0],e[1]=r*t[1],e[2]=r*t[2],e[3]=i*t[3],e[4]=i*t[4],e[5]=i*t[5],e[6]=t[6],e[7]=t[7],e[8]=t[8],e},l.fromMat2d=function(e,t){return e[0]=t[0],e[1]=t[1],e[2]=0,e[3]=t[2],e[4]=t[3],e[5]=0,e[6]=t[4],e[7]=t[5],e[8]=1,e},l.fromQuat=function(e,t){var n=t[0],r=t[1],i=t[2],s=t[3],o=n+n,u=r+r,a=i+i,f=n*o,l=n*u,c=n*a,h=r*u,p=r*a,d=i*a,v=s*o,m=s*u,g=s*a;return e[0]=1-(h+d),e[3]=l+g,e[6]=c-m,e[1]=l-g,e[4]=1-(f+d),e[7]=p+v,e[2]=c+m,e[5]=p-v,e[8]=1-(f+h),e},l.normalFromMat4=function(e,t){var n=t[0],r=t[1],i=t[2],s=t[3],o=t[4],u=t[5],a=t[6],f=t[7],l=t[8],c=t[9],h=t[10],p=t[11],d=t[12],v=t[13],m=t[14],g=t[15],y=n*u-r*o,b=n*a-i*o,w=n*f-s*o,E=r*a-i*u,S=r*f-s*u,x=i*f-s*a,T=l*v-c*d,N=l*m-h*d,C=l*g-p*d,k=c*m-h*v,L=c*g-p*v,A=h*g-p*m,O=y*A-b*L+w*k+E*C-S*N+x*T;return O?(O=1/O,e[0]=(u*A-a*L+f*k)*O,e[1]=(a*C-o*A-f*N)*O,e[2]=(o*L-u*C+f*T)*O,e[3]=(i*L-r*A-s*k)*O,e[4]=(n*A-i*C+s*N)*O,e[5]=(r*C-n*L-s*T)*O,e[6]=(v*x-m*S+g*E)*O,e[7]=(m*w-d*x-g*b)*O,e[8]=(d*S-v*w+g*y)*O,e):null},l.str=function(e){return\"mat3(\"+e[0]+\", \"+e[1]+\", \"+e[2]+\", \"+e[3]+\", \"+e[4]+\", \"+e[5]+\", \"+e[6]+\", \"+e[7]+\", \"+e[8]+\")\"},typeof e!=\"undefined\"&&(e.mat3=l);var c={};c.create=function(){var e=new n(16);return e[0]=1,e[1]=0,e[2]=0,e[3]=0,e[4]=0,e[5]=1,e[6]=0,e[7]=0,e[8]=0,e[9]=0,e[10]=1,e[11]=0,e[12]=0,e[13]=0,e[14]=0,e[15]=1,e},c.clone=function(e){var t=new n(16);return t[0]=e[0],t[1]=e[1],t[2]=e[2],t[3]=e[3],t[4]=e[4],t[5]=e[5],t[6]=e[6],t[7]=e[7],t[8]=e[8],t[9]=e[9],t[10]=e[10],t[11]=e[11],t[12]=e[12],t[13]=e[13],t[14]=e[14],t[15]=e[15],t},c.copy=function(e,t){return e[0]=t[0],e[1]=t[1],e[2]=t[2],e[3]=t[3],e[4]=t[4],e[5]=t[5],e[6]=t[6],e[7]=t[7],e[8]=t[8],e[9]=t[9],e[10]=t[10],e[11]=t[11],e[12]=t[12],e[13]=t[13],e[14]=t[14],e[15]=t[15],e},c.identity=function(e){return e[0]=1,e[1]=0,e[2]=0,e[3]=0,e[4]=0,e[5]=1,e[6]=0,e[7]=0,e[8]=0,e[9]=0,e[10]=1,e[11]=0,e[12]=0,e[13]=0,e[14]=0,e[15]=1,e},c.transpose=function(e,t){if(e===t){var n=t[1],r=t[2],i=t[3],s=t[6],o=t[7],u=t[11];e[1]=t[4],e[2]=t[8],e[3]=t[12],e[4]=n,e[6]=t[9],e[7]=t[13],e[8]=r,e[9]=s,e[11]=t[14],e[12]=i,e[13]=o,e[14]=u}else e[0]=t[0],e[1]=t[4],e[2]=t[8],e[3]=t[12],e[4]=t[1],e[5]=t[5],e[6]=t[9],e[7]=t[13],e[8]=t[2],e[9]=t[6],e[10]=t[10],e[11]=t[14],e[12]=t[3],e[13]=t[7],e[14]=t[11],e[15]=t[15];return e},c.invert=function(e,t){var n=t[0],r=t[1],i=t[2],s=t[3],o=t[4],u=t[5],a=t[6],f=t[7],l=t[8],c=t[9],h=t[10],p=t[11],d=t[12],v=t[13],m=t[14],g=t[15],y=n*u-r*o,b=n*a-i*o,w=n*f-s*o,E=r*a-i*u,S=r*f-s*u,x=i*f-s*a,T=l*v-c*d,N=l*m-h*d,C=l*g-p*d,k=c*m-h*v,L=c*g-p*v,A=h*g-p*m,O=y*A-b*L+w*k+E*C-S*N+x*T;return O?(O=1/O,e[0]=(u*A-a*L+f*k)*O,e[1]=(i*L-r*A-s*k)*O,e[2]=(v*x-m*S+g*E)*O,e[3]=(h*S-c*x-p*E)*O,e[4]=(a*C-o*A-f*N)*O,e[5]=(n*A-i*C+s*N)*O,e[6]=(m*w-d*x-g*b)*O,e[7]=(l*x-h*w+p*b)*O,e[8]=(o*L-u*C+f*T)*O,e[9]=(r*C-n*L-s*T)*O,e[10]=(d*S-v*w+g*y)*O,e[11]=(c*w-l*S-p*y)*O,e[12]=(u*N-o*k-a*T)*O,e[13]=(n*k-r*N+i*T)*O,e[14]=(v*b-d*E-m*y)*O,e[15]=(l*E-c*b+h*y)*O,e):null},c.adjoint=function(e,t){var n=t[0],r=t[1],i=t[2],s=t[3],o=t[4],u=t[5],a=t[6],f=t[7],l=t[8],c=t[9],h=t[10],p=t[11],d=t[12],v=t[13],m=t[14],g=t[15];return e[0]=u*(h*g-p*m)-c*(a*g-f*m)+v*(a*p-f*h),e[1]=-(r*(h*g-p*m)-c*(i*g-s*m)+v*(i*p-s*h)),e[2]=r*(a*g-f*m)-u*(i*g-s*m)+v*(i*f-s*a),e[3]=-(r*(a*p-f*h)-u*(i*p-s*h)+c*(i*f-s*a)),e[4]=-(o*(h*g-p*m)-l*(a*g-f*m)+d*(a*p-f*h)),e[5]=n*(h*g-p*m)-l*(i*g-s*m)+d*(i*p-s*h),e[6]=-(n*(a*g-f*m)-o*(i*g-s*m)+d*(i*f-s*a)),e[7]=n*(a*p-f*h)-o*(i*p-s*h)+l*(i*f-s*a),e[8]=o*(c*g-p*v)-l*(u*g-f*v)+d*(u*p-f*c),e[9]=-(n*(c*g-p*v)-l*(r*g-s*v)+d*(r*p-s*c)),e[10]=n*(u*g-f*v)-o*(r*g-s*v)+d*(r*f-s*u),e[11]=-(n*(u*p-f*c)-o*(r*p-s*c)+l*(r*f-s*u)),e[12]=-(o*(c*m-h*v)-l*(u*m-a*v)+d*(u*h-a*c)),e[13]=n*(c*m-h*v)-l*(r*m-i*v)+d*(r*h-i*c),e[14]=-(n*(u*m-a*v)-o*(r*m-i*v)+d*(r*a-i*u)),e[15]=n*(u*h-a*c)-o*(r*h-i*c)+l*(r*a-i*u),e},c.determinant=function(e){var t=e[0],n=e[1],r=e[2],i=e[3],s=e[4],o=e[5],u=e[6],a=e[7],f=e[8],l=e[9],c=e[10],h=e[11],p=e[12],d=e[13],v=e[14],m=e[15],g=t*o-n*s,y=t*u-r*s,b=t*a-i*s,w=n*u-r*o,E=n*a-i*o,S=r*a-i*u,x=f*d-l*p,T=f*v-c*p,N=f*m-h*p,C=l*v-c*d,k=l*m-h*d,L=c*m-h*v;return g*L-y*k+b*C+w*N-E*T+S*x},c.multiply=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=t[3],u=t[4],a=t[5],f=t[6],l=t[7],c=t[8],h=t[9],p=t[10],d=t[11],v=t[12],m=t[13],g=t[14],y=t[15],b=n[0],w=n[1],E=n[2],S=n[3];return e[0]=b*r+w*u+E*c+S*v,e[1]=b*i+w*a+E*h+S*m,e[2]=b*s+w*f+E*p+S*g,e[3]=b*o+w*l+E*d+S*y,b=n[4],w=n[5],E=n[6],S=n[7],e[4]=b*r+w*u+E*c+S*v,e[5]=b*i+w*a+E*h+S*m,e[6]=b*s+w*f+E*p+S*g,e[7]=b*o+w*l+E*d+S*y,b=n[8],w=n[9],E=n[10],S=n[11],e[8]=b*r+w*u+E*c+S*v,e[9]=b*i+w*a+E*h+S*m,e[10]=b*s+w*f+E*p+S*g,e[11]=b*o+w*l+E*d+S*y,b=n[12],w=n[13],E=n[14],S=n[15],e[12]=b*r+w*u+E*c+S*v,e[13]=b*i+w*a+E*h+S*m,e[14]=b*s+w*f+E*p+S*g,e[15]=b*o+w*l+E*d+S*y,e},c.mul=c.multiply,c.translate=function(e,t,n){var r=n[0],i=n[1],s=n[2],o,u,a,f,l,c,h,p,d,v,m,g;return t===e?(e[12]=t[0]*r+t[4]*i+t[8]*s+t[12],e[13]=t[1]*r+t[5]*i+t[9]*s+t[13],e[14]=t[2]*r+t[6]*i+t[10]*s+t[14],e[15]=t[3]*r+t[7]*i+t[11]*s+t[15]):(o=t[0],u=t[1],a=t[2],f=t[3],l=t[4],c=t[5],h=t[6],p=t[7],d=t[8],v=t[9],m=t[10],g=t[11],e[0]=o,e[1]=u,e[2]=a,e[3]=f,e[4]=l,e[5]=c,e[6]=h,e[7]=p,e[8]=d,e[9]=v,e[10]=m,e[11]=g,e[12]=o*r+l*i+d*s+t[12],e[13]=u*r+c*i+v*s+t[13],e[14]=a*r+h*i+m*s+t[14],e[15]=f*r+p*i+g*s+t[15]),e},c.scale=function(e,t,n){var r=n[0],i=n[1],s=n[2];return e[0]=t[0]*r,e[1]=t[1]*r,e[2]=t[2]*r,e[3]=t[3]*r,e[4]=t[4]*i,e[5]=t[5]*i,e[6]=t[6]*i,e[7]=t[7]*i,e[8]=t[8]*s,e[9]=t[9]*s,e[10]=t[10]*s,e[11]=t[11]*s,e[12]=t[12],e[13]=t[13],e[14]=t[14],e[15]=t[15],e},c.rotate=function(e,n,r,i){var s=i[0],o=i[1],u=i[2],a=Math.sqrt(s*s+o*o+u*u),f,l,c,h,p,d,v,m,g,y,b,w,E,S,x,T,N,C,k,L,A,O,M,_;return Math.abs(a)<t?null:(a=1/a,s*=a,o*=a,u*=a,f=Math.sin(r),l=Math.cos(r),c=1-l,h=n[0],p=n[1],d=n[2],v=n[3],m=n[4],g=n[5],y=n[6],b=n[7],w=n[8],E=n[9],S=n[10],x=n[11],T=s*s*c+l,N=o*s*c+u*f,C=u*s*c-o*f,k=s*o*c-u*f,L=o*o*c+l,A=u*o*c+s*f,O=s*u*c+o*f,M=o*u*c-s*f,_=u*u*c+l,e[0]=h*T+m*N+w*C,e[1]=p*T+g*N+E*C,e[2]=d*T+y*N+S*C,e[3]=v*T+b*N+x*C,e[4]=h*k+m*L+w*A,e[5]=p*k+g*L+E*A,e[6]=d*k+y*L+S*A,e[7]=v*k+b*L+x*A,e[8]=h*O+m*M+w*_,e[9]=p*O+g*M+E*_,e[10]=d*O+y*M+S*_,e[11]=v*O+b*M+x*_,n!==e&&(e[12]=n[12],e[13]=n[13],e[14]=n[14],e[15]=n[15]),e)},c.rotateX=function(e,t,n){var r=Math.sin(n),i=Math.cos(n),s=t[4],o=t[5],u=t[6],a=t[7],f=t[8],l=t[9],c=t[10],h=t[11];return t!==e&&(e[0]=t[0],e[1]=t[1],e[2]=t[2],e[3]=t[3],e[12]=t[12],e[13]=t[13],e[14]=t[14],e[15]=t[15]),e[4]=s*i+f*r,e[5]=o*i+l*r,e[6]=u*i+c*r,e[7]=a*i+h*r,e[8]=f*i-s*r,e[9]=l*i-o*r,e[10]=c*i-u*r,e[11]=h*i-a*r,e},c.rotateY=function(e,t,n){var r=Math.sin(n),i=Math.cos(n),s=t[0],o=t[1],u=t[2],a=t[3],f=t[8],l=t[9],c=t[10],h=t[11];return t!==e&&(e[4]=t[4],e[5]=t[5],e[6]=t[6],e[7]=t[7],e[12]=t[12],e[13]=t[13],e[14]=t[14],e[15]=t[15]),e[0]=s*i-f*r,e[1]=o*i-l*r,e[2]=u*i-c*r,e[3]=a*i-h*r,e[8]=s*r+f*i,e[9]=o*r+l*i,e[10]=u*r+c*i,e[11]=a*r+h*i,e},c.rotateZ=function(e,t,n){var r=Math.sin(n),i=Math.cos(n),s=t[0],o=t[1],u=t[2],a=t[3],f=t[4],l=t[5],c=t[6],h=t[7];return t!==e&&(e[8]=t[8],e[9]=t[9],e[10]=t[10],e[11]=t[11],e[12]=t[12],e[13]=t[13],e[14]=t[14],e[15]=t[15]),e[0]=s*i+f*r,e[1]=o*i+l*r,e[2]=u*i+c*r,e[3]=a*i+h*r,e[4]=f*i-s*r,e[5]=l*i-o*r,e[6]=c*i-u*r,e[7]=h*i-a*r,e},c.fromRotationTranslation=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=t[3],u=r+r,a=i+i,f=s+s,l=r*u,c=r*a,h=r*f,p=i*a,d=i*f,v=s*f,m=o*u,g=o*a,y=o*f;return e[0]=1-(p+v),e[1]=c+y,e[2]=h-g,e[3]=0,e[4]=c-y,e[5]=1-(l+v),e[6]=d+m,e[7]=0,e[8]=h+g,e[9]=d-m,e[10]=1-(l+p),e[11]=0,e[12]=n[0],e[13]=n[1],e[14]=n[2],e[15]=1,e},c.fromQuat=function(e,t){var n=t[0],r=t[1],i=t[2],s=t[3],o=n+n,u=r+r,a=i+i,f=n*o,l=n*u,c=n*a,h=r*u,p=r*a,d=i*a,v=s*o,m=s*u,g=s*a;return e[0]=1-(h+d),e[1]=l+g,e[2]=c-m,e[3]=0,e[4]=l-g,e[5]=1-(f+d),e[6]=p+v,e[7]=0,e[8]=c+m,e[9]=p-v,e[10]=1-(f+h),e[11]=0,e[12]=0,e[13]=0,e[14]=0,e[15]=1,e},c.frustum=function(e,t,n,r,i,s,o){var u=1/(n-t),a=1/(i-r),f=1/(s-o);return e[0]=s*2*u,e[1]=0,e[2]=0,e[3]=0,e[4]=0,e[5]=s*2*a,e[6]=0,e[7]=0,e[8]=(n+t)*u,e[9]=(i+r)*a,e[10]=(o+s)*f,e[11]=-1,e[12]=0,e[13]=0,e[14]=o*s*2*f,e[15]=0,e},c.perspective=function(e,t,n,r,i){var s=1/Math.tan(t/2),o=1/(r-i);return e[0]=s/n,e[1]=0,e[2]=0,e[3]=0,e[4]=0,e[5]=s,e[6]=0,e[7]=0,e[8]=0,e[9]=0,e[10]=(i+r)*o,e[11]=-1,e[12]=0,e[13]=0,e[14]=2*i*r*o,e[15]=0,e},c.ortho=function(e,t,n,r,i,s,o){var u=1/(t-n),a=1/(r-i),f=1/(s-o);return e[0]=-2*u,e[1]=0,e[2]=0,e[3]=0,e[4]=0,e[5]=-2*a,e[6]=0,e[7]=0,e[8]=0,e[9]=0,e[10]=2*f,e[11]=0,e[12]=(t+n)*u,e[13]=(i+r)*a,e[14]=(o+s)*f,e[15]=1,e},c.lookAt=function(e,n,r,i){var s,o,u,a,f,l,h,p,d,v,m=n[0],g=n[1],y=n[2],b=i[0],w=i[1],E=i[2],S=r[0],x=r[1],T=r[2];return Math.abs(m-S)<t&&Math.abs(g-x)<t&&Math.abs(y-T)<t?c.identity(e):(h=m-S,p=g-x,d=y-T,v=1/Math.sqrt(h*h+p*p+d*d),h*=v,p*=v,d*=v,s=w*d-E*p,o=E*h-b*d,u=b*p-w*h,v=Math.sqrt(s*s+o*o+u*u),v?(v=1/v,s*=v,o*=v,u*=v):(s=0,o=0,u=0),a=p*u-d*o,f=d*s-h*u,l=h*o-p*s,v=Math.sqrt(a*a+f*f+l*l),v?(v=1/v,a*=v,f*=v,l*=v):(a=0,f=0,l=0),e[0]=s,e[1]=a,e[2]=h,e[3]=0,e[4]=o,e[5]=f,e[6]=p,e[7]=0,e[8]=u,e[9]=l,e[10]=d,e[11]=0,e[12]=-(s*m+o*g+u*y),e[13]=-(a*m+f*g+l*y),e[14]=-(h*m+p*g+d*y),e[15]=1,e)},c.str=function(e){return\"mat4(\"+e[0]+\", \"+e[1]+\", \"+e[2]+\", \"+e[3]+\", \"+e[4]+\", \"+e[5]+\", \"+e[6]+\", \"+e[7]+\", \"+e[8]+\", \"+e[9]+\", \"+e[10]+\", \"+e[11]+\", \"+e[12]+\", \"+e[13]+\", \"+e[14]+\", \"+e[15]+\")\"},typeof e!=\"undefined\"&&(e.mat4=c);var h={};h.create=function(){var e=new n(4);return e[0]=0,e[1]=0,e[2]=0,e[3]=1,e},h.rotationTo=function(){var e=o.create(),t=o.fromValues(1,0,0),n=o.fromValues(0,1,0);return function(r,i,s){var u=o.dot(i,s);return u<-0.999999?(o.cross(e,t,i),o.length(e)<1e-6&&o.cross(e,n,i),o.normalize(e,e),h.setAxisAngle(r,e,Math.PI),r):u>.999999?(r[0]=0,r[1]=0,r[2]=0,r[3]=1,r):(o.cross(e,i,s),r[0]=e[0],r[1]=e[1],r[2]=e[2],r[3]=1+u,h.normalize(r,r))}}(),h.setAxes=function(){var e=l.create();return function(t,n,r,i){return e[0]=r[0],e[3]=r[1],e[6]=r[2],e[1]=i[0],e[4]=i[1],e[7]=i[2],e[2]=n[0],e[5]=n[1],e[8]=n[2],h.normalize(t,h.fromMat3(t,e))}}(),h.clone=u.clone,h.fromValues=u.fromValues,h.copy=u.copy,h.set=u.set,h.identity=function(e){return e[0]=0,e[1]=0,e[2]=0,e[3]=1,e},h.setAxisAngle=function(e,t,n){n*=.5;var r=Math.sin(n);return e[0]=r*t[0],e[1]=r*t[1],e[2]=r*t[2],e[3]=Math.cos(n),e},h.add=u.add,h.multiply=function(e,t,n){var r=t[0],i=t[1],s=t[2],o=t[3],u=n[0],a=n[1],f=n[2],l=n[3];return e[0]=r*l+o*u+i*f-s*a,e[1]=i*l+o*a+s*u-r*f,e[2]=s*l+o*f+r*a-i*u,e[3]=o*l-r*u-i*a-s*f,e},h.mul=h.multiply,h.scale=u.scale,h.rotateX=function(e,t,n){n*=.5;var r=t[0],i=t[1],s=t[2],o=t[3],u=Math.sin(n),a=Math.cos(n);return e[0]=r*a+o*u,e[1]=i*a+s*u,e[2]=s*a-i*u,e[3]=o*a-r*u,e},h.rotateY=function(e,t,n){n*=.5;var r=t[0],i=t[1],s=t[2],o=t[3],u=Math.sin(n),a=Math.cos(n);return e[0]=r*a-s*u,e[1]=i*a+o*u,e[2]=s*a+r*u,e[3]=o*a-i*u,e},h.rotateZ=function(e,t,n){n*=.5;var r=t[0],i=t[1],s=t[2],o=t[3],u=Math.sin(n),a=Math.cos(n);return e[0]=r*a+i*u,e[1]=i*a-r*u,e[2]=s*a+o*u,e[3]=o*a-s*u,e},h.calculateW=function(e,t){var n=t[0],r=t[1],i=t[2];return e[0]=n,e[1]=r,e[2]=i,e[3]=-Math.sqrt(Math.abs(1-n*n-r*r-i*i)),e},h.dot=u.dot,h.lerp=u.lerp,h.slerp=function(e,t,n,r){var i=t[0],s=t[1],o=t[2],u=t[3],a=n[0],f=n[1],l=n[2],c=n[3],h,p,d,v,m;return p=i*a+s*f+o*l+u*c,p<0&&(p=-p,a=-a,f=-f,l=-l,c=-c),1-p>1e-6?(h=Math.acos(p),d=Math.sin(h),v=Math.sin((1-r)*h)/d,m=Math.sin(r*h)/d):(v=1-r,m=r),e[0]=v*i+m*a,e[1]=v*s+m*f,e[2]=v*o+m*l,e[3]=v*u+m*c,e},h.invert=function(e,t){var n=t[0],r=t[1],i=t[2],s=t[3],o=n*n+r*r+i*i+s*s,u=o?1/o:0;return e[0]=-n*u,e[1]=-r*u,e[2]=-i*u,e[3]=s*u,e},h.conjugate=function(e,t){return e[0]=-t[0],e[1]=-t[1],e[2]=-t[2],e[3]=t[3],e},h.length=u.length,h.len=h.length,h.squaredLength=u.squaredLength,h.sqrLen=h.squaredLength,h.normalize=u.normalize,h.fromMat3=function(){var e=typeof Int8Array!=\"undefined\"?new Int8Array([1,2,0]):[1,2,0];return function(t,n){var r=n[0]+n[4]+n[8],i;if(r>0)i=Math.sqrt(r+1),t[3]=.5*i,i=.5/i,t[0]=(n[7]-n[5])*i,t[1]=(n[2]-n[6])*i,t[2]=(n[3]-n[1])*i;else{var s=0;n[4]>n[0]&&(s=1),n[8]>n[s*3+s]&&(s=2);var o=e[s],u=e[o];i=Math.sqrt(n[s*3+s]-n[o*3+o]-n[u*3+u]+1),t[s]=.5*i,i=.5/i,t[3]=(n[u*3+o]-n[o*3+u])*i,t[o]=(n[o*3+s]+n[s*3+o])*i,t[u]=(n[u*3+s]+n[s*3+u])*i}return t}}(),h.str=function(e){return\"quat(\"+e[0]+\", \"+e[1]+\", \"+e[2]+\", \"+e[3]+\")\"},typeof e!=\"undefined\"&&(e.quat=h)}(t.exports)})(this);\n","'use strict';\n\nif (typeof window === 'undefined') {\n    new (require('./source/worker.js'))(self);\n} else {\n    // jshint -W079\n    var mapboxgl = module.exports = window.mapboxgl = {};\n\n    mapboxgl.Map = require('./ui/map.js');\n    mapboxgl.Navigation = require('./ui/control/navigation.js');\n    mapboxgl.Attribution = require('./ui/control/attribution.js');\n\n    mapboxgl.Source = require('./source/source');\n    mapboxgl.GeoJSONSource = require('./source/geojsonsource');\n    mapboxgl.VideoSource = require('./source/videosource');\n\n    mapboxgl.Style = require('./style/style.js');\n\n    mapboxgl.LatLng = require('./geo/latlng.js');\n    mapboxgl.LatLngBounds = require('./geo/latlngbounds.js');\n    mapboxgl.Point = require('point-geometry');\n\n    mapboxgl.Evented = require('./util/evented.js');\n    mapboxgl.util = require('./util/util.js');\n\n    var browser = require('./util/browser.js');\n    mapboxgl.util.supported = browser.supported;\n\n    var ajax = require('./util/ajax.js');\n    mapboxgl.util.getJSON = ajax.getJSON;\n    mapboxgl.util.getArrayBuffer = ajax.getArrayBuffer;\n\n    var config = require('./util/config.js');\n    mapboxgl.config = config;\n\n    Object.defineProperty(mapboxgl, 'accessToken', {\n        get: function() { return config.ACCESS_TOKEN; },\n        set: function(token) { config.ACCESS_TOKEN = token; }\n    });\n}\n","'use strict';\n\nvar mat3 = require('../lib/glmatrix.js').mat3;\n\nmodule.exports = drawBackground;\n\nfunction drawBackground(gl, painter, bucket, layerStyle, posMatrix, params, imageSprite) {\n    var color = layerStyle['background-color'];\n    var image = layerStyle['background-image'];\n    var opacity = layerStyle['background-opacity'] || 1;\n    var shader;\n\n    if (image) {\n        // Draw texture fill\n        var imagePos = imageSprite.getPosition(image, true);\n        if (!imagePos) return;\n\n        shader = painter.patternShader;\n        gl.switchShader(shader, posMatrix);\n        gl.uniform1i(shader.u_image, 0);\n        gl.uniform2fv(shader.u_pattern_tl, imagePos.tl);\n        gl.uniform2fv(shader.u_pattern_br, imagePos.br);\n        gl.uniform1f(shader.u_mix, painter.transform.zoomFraction);\n        gl.uniform1f(shader.u_opacity, opacity);\n\n        var transform = painter.transform;\n        var size = imagePos.size;\n        var center = transform.locationCoordinate(transform.center);\n        var scale = 1 / Math.pow(2, transform.zoomFraction);\n        var matrix = mat3.create();\n\n        mat3.scale(matrix, matrix, [1 / size[0], 1 / size[1], 1]);\n        mat3.translate(matrix, matrix, [\n            (center.column * transform.tileSize) % size[0],\n            (center.row    * transform.tileSize) % size[1],\n            0\n        ]);\n        mat3.rotate(matrix, matrix, -transform.angle);\n        mat3.scale(matrix, matrix, [\n            scale * transform.width  / 2,\n           -scale * transform.height / 2,\n            1\n        ]);\n\n        gl.uniformMatrix3fv(shader.u_patternmatrix, false, matrix);\n\n        imageSprite.bind(gl, true);\n\n    } else {\n        // Draw filling rectangle.\n        shader = painter.fillShader;\n        gl.switchShader(shader, params.padded || posMatrix);\n        gl.uniform4fv(shader.u_color, color);\n    }\n\n    gl.disable(gl.STENCIL_TEST);\n    gl.bindBuffer(gl.ARRAY_BUFFER, painter.backgroundBuffer);\n    gl.vertexAttribPointer(shader.a_pos, painter.backgroundBuffer.itemSize, gl.SHORT, false, 0, 0);\n    gl.drawArrays(gl.TRIANGLE_STRIP, 0, painter.backgroundBuffer.itemCount);\n    gl.enable(gl.STENCIL_TEST);\n\n    gl.stencilMask(0x00);\n    gl.stencilFunc(gl.EQUAL, 0x80, 0x80);\n}\n","'use strict';\n\nmodule.exports = drawComposited;\n\nfunction drawComposited (gl, painter, buckets, layerStyle, params, style, layer) {\n    var texture = painter.namedRenderTextures[layer.id];\n    if (!texture) return console.warn('missing render texture ' + layer.id);\n\n    gl.disable(gl.STENCIL_TEST);\n    gl.stencilMask(0x00);\n\n    gl.switchShader(painter.compositeShader, painter.identityMatrix);\n    gl.activeTexture(gl.TEXTURE0);\n    gl.bindTexture(gl.TEXTURE_2D, texture);\n    gl.uniform1i(painter.compositeShader.u_image, 0);\n\n    gl.uniform1f(painter.compositeShader.u_opacity, layerStyle['composite-opacity']);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, painter.backgroundBuffer);\n    gl.vertexAttribPointer(painter.compositeShader.a_pos, 2, gl.SHORT, false, 0, 0);\n    gl.drawArrays(gl.TRIANGLE_STRIP, 0, 4);\n\n    gl.enable(gl.STENCIL_TEST);\n\n    painter.freeRenderTexture(layer.id);\n}\n","'use strict';\n\nvar textVertices = require('../lib/debugtext.js');\nvar browser = require('../util/browser.js');\n\nmodule.exports = drawDebug;\n\nfunction drawDebug(gl, painter, tile, params) {\n    // Blend to the front, not the back.\n    gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA);\n\n    gl.switchShader(painter.debugShader, painter.tile.posMatrix, painter.tile.exMatrix);\n\n    // draw bounding rectangle\n    gl.bindBuffer(gl.ARRAY_BUFFER, painter.debugBuffer);\n    gl.vertexAttribPointer(painter.debugShader.a_pos, painter.debugBuffer.itemSize, gl.SHORT, false, 0, 0);\n    gl.uniform4f(painter.debugShader.u_color, 1, 0, 0, 1);\n    gl.lineWidth(4);\n    gl.drawArrays(gl.LINE_STRIP, 0, painter.debugBuffer.itemCount);\n\n    // draw tile coordinate\n    var coord = params.z + '/' + params.x + '/' + params.y;\n\n    var vertices = textVertices(coord, 50, 200, 5);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, painter.debugTextBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, new Int16Array(vertices), gl.STREAM_DRAW);\n    gl.vertexAttribPointer(painter.debugShader.a_pos, painter.debugTextBuffer.itemSize, gl.SHORT, false, 0, 0);\n    gl.lineWidth(8 * browser.devicePixelRatio);\n    gl.uniform4f(painter.debugShader.u_color, 1, 1, 1, 1);\n    gl.drawArrays(gl.LINES, 0, vertices.length / painter.debugTextBuffer.itemSize);\n    gl.lineWidth(2 * browser.devicePixelRatio);\n    gl.uniform4f(painter.debugShader.u_color, 0, 0, 0, 1);\n    gl.drawArrays(gl.LINES, 0, vertices.length / painter.debugTextBuffer.itemSize);\n\n    // Revert blending mode to blend to the back.\n    gl.blendFunc(gl.ONE_MINUS_DST_ALPHA, gl.ONE);\n}\n","'use strict';\n\nvar browser = require('../util/browser.js');\nvar mat3 = require('../lib/glmatrix.js').mat3;\n\nmodule.exports = drawFill;\n\nfunction drawFill(gl, painter, bucket, layerStyle, posMatrix, params, imageSprite) {\n\n    var translatedPosMatrix = painter.translateMatrix(posMatrix, params.z, layerStyle['fill-translate'], layerStyle['fill-translate-anchor']);\n\n    var color = layerStyle['fill-color'];\n\n    var vertex, elements, group, count;\n\n    // Draw the stencil mask.\n\n    // We're only drawing to the first seven bits (== support a maximum of\n    // 127 overlapping polygons in one place before we get rendering errors).\n    gl.stencilMask(0x3F);\n    gl.clear(gl.STENCIL_BUFFER_BIT);\n\n    // Draw front facing triangles. Wherever the 0x80 bit is 1, we are\n    // increasing the lower 7 bits by one if the triangle is a front-facing\n    // triangle. This means that all visible polygons should be in CCW\n    // orientation, while all holes (see below) are in CW orientation.\n    gl.stencilFunc(gl.NOTEQUAL, 0x80, 0x80);\n\n    // When we do a nonzero fill, we count the number of times a pixel is\n    // covered by a counterclockwise polygon, and subtract the number of\n    // times it is \"uncovered\" by a clockwise polygon.\n    gl.stencilOpSeparate(gl.FRONT, gl.INCR_WRAP, gl.KEEP, gl.KEEP);\n    gl.stencilOpSeparate(gl.BACK, gl.DECR_WRAP, gl.KEEP, gl.KEEP);\n\n    // When drawing a shape, we first draw all shapes to the stencil buffer\n    // and incrementing all areas where polygons are\n    gl.colorMask(false, false, false, false);\n\n    // Draw the actual triangle fan into the stencil buffer.\n    gl.switchShader(painter.fillShader, translatedPosMatrix, painter.tile.exMatrix);\n\n    // Draw all buffers\n    vertex = bucket.buffers.fillVertex;\n    vertex.bind(gl);\n    elements = bucket.buffers.fillElement;\n    elements.bind(gl);\n\n    var offset, elementOffset;\n    for (var i = 0; i < bucket.elementGroups.groups.length; i++) {\n        group = bucket.elementGroups.groups[i];\n        offset = group.vertexStartIndex * vertex.itemSize;\n        gl.vertexAttribPointer(painter.fillShader.a_pos, 2, gl.SHORT, false, 4, offset + 0);\n\n        count = group.elementLength * 3;\n        elementOffset = group.elementStartIndex * elements.itemSize;\n        gl.drawElements(gl.TRIANGLES, count, gl.UNSIGNED_SHORT, elementOffset);\n    }\n\n    // Now that we have the stencil mask in the stencil buffer, we can start\n    // writing to the color buffer.\n    gl.colorMask(true, true, true, true);\n\n    // From now on, we don't want to update the stencil buffer anymore.\n    gl.stencilOp(gl.KEEP, gl.KEEP, gl.KEEP);\n    gl.stencilMask(0x0);\n\n    var strokeColor = layerStyle['fill-outline-color'];\n\n    // Because we're drawing top-to-bottom, and we update the stencil mask\n    // below, we have to draw the outline first (!)\n    if (layerStyle['fill-antialias'] === true && params.antialiasing && !(layerStyle['fill-image'] && !strokeColor)) {\n        gl.switchShader(painter.outlineShader, translatedPosMatrix, painter.tile.exMatrix);\n        gl.lineWidth(2 * browser.devicePixelRatio);\n\n        if (strokeColor) {\n            // If we defined a different color for the fill outline, we are\n            // going to ignore the bits in 0x3F and just care about the global\n            // clipping mask.\n            gl.stencilFunc(gl.EQUAL, 0x80, 0x80);\n        } else {\n            // Otherwise, we only want to draw the antialiased parts that are\n            // *outside* the current shape. This is important in case the fill\n            // or stroke color is translucent. If we wouldn't clip to outside\n            // the current shape, some pixels from the outline stroke overlapped\n            // the (non-antialiased) fill.\n            gl.stencilFunc(gl.EQUAL, 0x80, 0xBF);\n        }\n\n        gl.uniform2f(painter.outlineShader.u_world, gl.drawingBufferWidth, gl.drawingBufferHeight);\n        gl.uniform4fv(painter.outlineShader.u_color, strokeColor ? strokeColor : color);\n\n        // Draw all buffers\n        vertex = bucket.buffers.fillVertex;\n        elements = bucket.buffers.outlineElement;\n        elements.bind(gl);\n\n        for (var k = 0; k < bucket.elementGroups.groups.length; k++) {\n            group = bucket.elementGroups.groups[k];\n            offset = group.vertexStartIndex * vertex.itemSize;\n            gl.vertexAttribPointer(painter.outlineShader.a_pos, 2, gl.SHORT, false, 4, offset + 0);\n\n            count = group.secondElementLength * 2;\n            elementOffset = group.secondElementStartIndex * elements.itemSize;\n            gl.drawElements(gl.LINES, count, gl.UNSIGNED_SHORT, elementOffset);\n        }\n    }\n\n    var image = layerStyle['fill-image'];\n    var opacity = layerStyle['fill-opacity'] || 1;\n    var shader;\n\n    if (image) {\n        // Draw texture fill\n        var imagePos = imageSprite.getPosition(image, true);\n        if (!imagePos) return;\n\n        shader = painter.patternShader;\n        gl.switchShader(shader, posMatrix);\n        gl.uniform1i(shader.u_image, 0);\n        gl.uniform2fv(shader.u_pattern_tl, imagePos.tl);\n        gl.uniform2fv(shader.u_pattern_br, imagePos.br);\n        gl.uniform1f(shader.u_mix, painter.transform.zoomFraction);\n        gl.uniform1f(shader.u_opacity, opacity);\n\n        var factor = 8 / Math.pow(2, painter.transform.tileZoom - params.z);\n\n        var matrix = mat3.create();\n        mat3.scale(matrix, matrix, [\n            1 / (imagePos.size[0] * factor),\n            1 / (imagePos.size[1] * factor),\n            1, 1\n        ]);\n\n        gl.uniformMatrix3fv(shader.u_patternmatrix, false, matrix);\n\n        imageSprite.bind(gl, true);\n\n    } else {\n        // Draw filling rectangle.\n        shader = painter.fillShader;\n        gl.switchShader(shader, params.padded || posMatrix);\n        gl.uniform4fv(shader.u_color, color);\n    }\n\n    // Only draw regions that we marked\n    gl.stencilFunc(gl.NOTEQUAL, 0x0, 0x3F);\n    gl.bindBuffer(gl.ARRAY_BUFFER, painter.tileExtentBuffer);\n    gl.vertexAttribPointer(shader.a_pos, painter.tileExtentBuffer.itemSize, gl.SHORT, false, 0, 0);\n    gl.drawArrays(gl.TRIANGLE_STRIP, 0, painter.tileExtentBuffer.itemCount);\n\n    gl.stencilMask(0x00);\n    gl.stencilFunc(gl.EQUAL, 0x80, 0x80);\n}\n","'use strict';\n\nvar browser = require('../util/browser.js');\n\nmodule.exports = function drawLine(gl, painter, bucket, layerStyle, posMatrix, params, imageSprite) {\n\n    posMatrix = painter.translateMatrix(posMatrix, params.z, layerStyle['line-translate'], layerStyle['line-translate-anchor']);\n\n    // don't draw zero-width lines\n    if (layerStyle['line-width'] <= 0) return;\n\n    var gamma = 1;\n    var antialiasing = gamma / browser.devicePixelRatio;\n\n    var lineOffset = layerStyle['line-offset'] / 2;\n    var inset = Math.max(-1, lineOffset - layerStyle['line-width'] / 2 - antialiasing / 2) + 1;\n    var outset = lineOffset + layerStyle['line-width'] / 2 + antialiasing / 2;\n\n    var imagePos = layerStyle['line-image'] && imageSprite.getPosition(layerStyle['line-image']);\n    var shader;\n\n    if (imagePos) {\n        var factor = 8 / Math.pow(2, painter.transform.tileZoom - params.z);\n\n        imageSprite.bind(gl, true);\n\n        //factor = Math.pow(2, 4 - painter.transform.tileZoom + params.z);\n        shader = painter.linepatternShader;\n        gl.switchShader(shader, posMatrix, painter.tile.exMatrix);\n        gl.uniform2fv(shader.u_pattern_size, [imagePos.size[0] * factor, imagePos.size[1] ]);\n        gl.uniform2fv(shader.u_pattern_tl, imagePos.tl);\n        gl.uniform2fv(shader.u_pattern_br, imagePos.br);\n        gl.uniform1f(shader.u_fade, painter.transform.zoomFraction);\n\n    } else {\n        shader = painter.lineShader;\n        gl.switchShader(shader, posMatrix, painter.tile.exMatrix);\n        gl.uniform2fv(shader.u_dasharray, layerStyle['line-dasharray']);\n        gl.uniform4fv(shader.u_color, layerStyle['line-color']);\n    }\n\n    var tilePixelRatio = painter.transform.scale / (1 << params.z) / 8;\n    gl.uniform2fv(shader.u_linewidth, [ outset, inset ]);\n    gl.uniform1f(shader.u_ratio, tilePixelRatio);\n    gl.uniform1f(shader.u_blur, layerStyle['line-blur'] + antialiasing);\n\n\n    var vertex = bucket.buffers.lineVertex;\n    vertex.bind(gl);\n    var element = bucket.buffers.lineElement;\n    element.bind(gl);\n\n    var groups = bucket.elementGroups.groups;\n    for (var i = 0; i < groups.length; i++) {\n        var group = groups[i];\n        var offset = group.vertexStartIndex * vertex.itemSize;\n        gl.vertexAttribPointer(shader.a_pos, 4, gl.SHORT, false, 8, offset + 0);\n        gl.vertexAttribPointer(shader.a_extrude, 2, gl.BYTE, false, 8, offset + 6);\n        gl.vertexAttribPointer(shader.a_linesofar, 2, gl.SHORT, false, 8, offset + 4);\n\n        var count = group.elementLength * 3;\n        var elementOffset = group.elementStartIndex * element.itemSize;\n        gl.drawElements(gl.TRIANGLES, count, gl.UNSIGNED_SHORT, elementOffset);\n    }\n\n};\n","'use strict';\n\nvar TileCoord = require('../source/tilecoord.js');\nvar PrerenderedTexture = require('./prerendered.js');\nvar mat4 = require('../lib/glmatrix.js').mat4;\n\nmodule.exports = drawRaster;\n\nfunction drawRaster(gl, painter, bucket, layerStyle, params, style, layer, tile) {\n\n    if (layer && layer.layers) {\n\n        if (!bucket.prerendered) {\n            bucket.prerendered = new PrerenderedTexture(gl, bucket.info, painter);\n            bucket.prerendered.bindFramebuffer();\n\n            gl.clearStencil(0x80);\n            gl.stencilMask(0xFF);\n            gl.clear(gl.STENCIL_BUFFER_BIT | gl.COLOR_BUFFER_BIT);\n            gl.stencilMask(0x00);\n\n            gl.viewport(0, 0, bucket.prerendered.size, bucket.prerendered.size);\n\n            var buffer = bucket.prerendered.buffer * 4096;\n\n            var matrix = mat4.create();\n            mat4.ortho(matrix, -buffer, 4096 + buffer, -4096 - buffer, buffer, 0, 1);\n            mat4.translate(matrix, matrix, [0, -4096, 0]);\n\n            params.padded = mat4.create();\n            mat4.ortho(params.padded, 0, 4096, -4096, 0, 0, 1);\n            mat4.translate(params.padded, params.padded, [0, -4096, 0]);\n\n            painter.draw(tile, style, layer.layers, params, matrix);\n\n            delete params.padded;\n\n            if (bucket.info['raster-blur'] > 0) {\n                bucket.prerendered.blur(painter, bucket.info['raster-blur']);\n            }\n\n            bucket.prerendered.unbindFramebuffer();\n            gl.viewport(0, 0, painter.width, painter.height);\n        }\n\n    }\n\n    var texture = bucket.tile ? bucket.tile : bucket.prerendered;\n\n    gl.disable(gl.STENCIL_TEST);\n\n    var shader = painter.rasterShader;\n    gl.switchShader(shader, painter.tile.posMatrix, painter.tile.exMatrix);\n\n    // color parameters\n    gl.uniform1f(shader.u_brightness_low, layerStyle['raster-brightness'][0]);\n    gl.uniform1f(shader.u_brightness_high, layerStyle['raster-brightness'][1]);\n    gl.uniform1f(shader.u_saturation_factor, saturationFactor(layerStyle['raster-saturation']));\n    gl.uniform1f(shader.u_contrast_factor, contrastFactor(layerStyle['raster-contrast']));\n    gl.uniform3fv(shader.u_spin_weights, spinWeights(layerStyle['raster-hue-rotate']));\n\n\n    var parentTile, opacities;\n    if (layer && layer.layers) {\n        parentTile = null;\n        opacities = [layerStyle['raster-opacity'], 0];\n    } else {\n        parentTile = findParent(texture);\n        opacities = getOpacities(texture, parentTile);\n    }\n    var parentScaleBy, parentTL;\n\n    gl.activeTexture(gl.TEXTURE0);\n    texture.bind(gl);\n\n    if (parentTile) {\n        gl.activeTexture(gl.TEXTURE1);\n        parentTile.bind(gl);\n\n        var tilePos = TileCoord.fromID(texture.id);\n        var parentPos = parentTile && TileCoord.fromID(parentTile.id);\n        parentScaleBy = Math.pow(2, parentPos.z - tilePos.z);\n        parentTL = [tilePos.x * parentScaleBy % 1, tilePos.y * parentScaleBy % 1];\n    } else {\n        opacities[1] = 0;\n    }\n\n    var bufferScale = bucket.prerendered ? (4096 * (1 + 2 * bucket.prerendered.buffer)) / 4096 : 1;\n\n    // cross-fade parameters\n    gl.uniform2fv(shader.u_tl_parent, parentTL || [0, 0]);\n    gl.uniform1f(shader.u_scale_parent, parentScaleBy || 1);\n    gl.uniform1f(shader.u_buffer_scale, bufferScale);\n    gl.uniform1f(shader.u_opacity0, opacities[0]);\n    gl.uniform1f(shader.u_opacity1, opacities[1]);\n    gl.uniform1i(shader.u_image0, 0);\n    gl.uniform1i(shader.u_image1, 1);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, texture.boundsBuffer || painter.tileExtentBuffer);\n\n    gl.vertexAttribPointer(shader.a_pos,         2, gl.SHORT, false, 8, 0);\n    gl.vertexAttribPointer(shader.a_texture_pos, 2, gl.SHORT, false, 8, 4);\n    gl.drawArrays(gl.TRIANGLE_STRIP, 0, 4);\n\n    gl.enable(gl.STENCIL_TEST);\n}\n\nfunction findParent(tile) {\n    var source = tile.source;\n    if (!source) return;\n    var parentTiles = {};\n    source._findLoadedParent(tile.id, source.options.minZoom, parentTiles);\n    return source.tiles[Object.keys(parentTiles)[0]];\n}\n\nfunction clamp(n, min, max) {\n    return Math.max(min, Math.min(max, n));\n}\n\nfunction spinWeights(angle) {\n    angle *= Math.PI / 180;\n    var s = Math.sin(angle);\n    var c = Math.cos(angle);\n    return [\n        (2 * c + 1) / 3,\n        (-Math.sqrt(3) * s - c + 1) / 3,\n        (Math.sqrt(3) * s - c + 1) / 3\n    ];\n}\n\nfunction contrastFactor(contrast) {\n    return contrast > 0 ?\n        1 / (1 - contrast) :\n        1 + contrast;\n}\n\nfunction saturationFactor(saturation) {\n    return saturation > 0 ?\n        1 - 1 / (1.001 - saturation) :\n        -saturation;\n}\n\nfunction getOpacities(tile, parentTile) {\n    if (!tile.source) return [1, 0];\n\n    var now = new Date().getTime();\n    var fadeDuration = tile.source.map.style.rasterFadeDuration;\n\n    var sinceTile = (now - tile.timeAdded) / fadeDuration;\n    var sinceParent = parentTile ? (now - parentTile.timeAdded) / fadeDuration : -1;\n\n    var tilePos = TileCoord.fromID(tile.id);\n    var parentPos = parentTile && TileCoord.fromID(parentTile.id);\n\n    var idealZ = tile.source._coveringZoomLevel(tile.source._getZoom());\n    var parentFurther = parentTile ? Math.abs(parentPos.z - idealZ) > Math.abs(tilePos.z - idealZ) : false;\n\n    var opacity = [];\n    if (!parentTile || parentFurther) {\n        // if no parent or parent is older\n        opacity[0] = clamp(sinceTile, 0, 1);\n        opacity[1] = 1 - opacity[0];\n    } else {\n        // parent is younger, zooming out\n        opacity[0] = clamp(1 - sinceParent, 0, 1);\n        opacity[1] = 1 - opacity[0];\n    }\n\n    return opacity;\n}\n","'use strict';\n\nvar browser = require('../util/browser.js');\nvar mat4 = require('../lib/glmatrix.js').mat4;\n\nmodule.exports = drawSymbols;\n\nfunction drawSymbols(gl, painter, bucket, layerStyle, posMatrix, params, imageSprite) {\n    gl.disable(gl.STENCIL_TEST);\n    if (bucket.elementGroups.text.groups.length) {\n        drawSymbol(gl, painter, bucket, layerStyle, posMatrix, params, imageSprite, 'text');\n    }\n    if (bucket.elementGroups.icon.groups.length) {\n        drawSymbol(gl, painter, bucket, layerStyle, posMatrix, params, imageSprite, 'icon');\n    }\n    gl.enable(gl.STENCIL_TEST);\n}\n\nvar defaultSizes = {\n    icon: 1,\n    text: 24\n};\n\nfunction drawSymbol(gl, painter, bucket, layerStyle, posMatrix, params, imageSprite, prefix) {\n\n    posMatrix = painter.translateMatrix(posMatrix, params.z, layerStyle[prefix + '-translate'], layerStyle[prefix + '-translate-anchor']);\n\n    var info = bucket.info;\n\n    var exMatrix = mat4.clone(painter.projectionMatrix);\n    var alignedWithMap = info[prefix + '-rotation-alignment'] === 'map';\n    var angleOffset = (alignedWithMap ? painter.transform.angle : 0);\n\n    if (angleOffset) {\n        mat4.rotateZ(exMatrix, exMatrix, angleOffset);\n    }\n\n    // If layerStyle.size > info[prefix + '-max-size'] then labels may collide\n    var fontSize = layerStyle[prefix + '-size'] || info[prefix + '-max-size'];\n    var fontScale = fontSize / defaultSizes[prefix];\n    mat4.scale(exMatrix, exMatrix, [ fontScale, fontScale, 1 ]);\n\n    var text = prefix === 'text';\n    var sdf = text || bucket.elementGroups.sdfIcons;\n    var shader, buffer, texsize;\n\n    if (!text && !imageSprite.loaded())\n        return;\n\n    gl.activeTexture(gl.TEXTURE0);\n\n    if (sdf) {\n        shader = painter.sdfShader;\n    } else {\n        shader = painter.iconShader;\n    }\n\n    if (text) {\n        painter.glyphAtlas.updateTexture(gl);\n        buffer = bucket.buffers.glyphVertex;\n        texsize = [painter.glyphAtlas.width / 4, painter.glyphAtlas.height / 4];\n    } else {\n        imageSprite.bind(gl, alignedWithMap || params.rotating || params.zooming || fontScale != 1 || sdf);\n        buffer = bucket.buffers.iconVertex;\n        texsize = [imageSprite.img.width, imageSprite.img.height];\n    }\n\n    gl.switchShader(shader, posMatrix, exMatrix);\n    gl.uniform1i(shader.u_texture, 0);\n    gl.uniform2fv(shader.u_texsize, texsize);\n\n    buffer.bind(gl);\n\n    var ubyte = gl.UNSIGNED_BYTE;\n\n    var stride = text ? 16 : 20;\n\n    gl.vertexAttribPointer(shader.a_pos,          2, gl.SHORT, false, stride, 0);\n    gl.vertexAttribPointer(shader.a_offset,       2, gl.SHORT, false, stride, 4);\n    gl.vertexAttribPointer(shader.a_labelminzoom, 1, ubyte,    false, stride, 8);\n    gl.vertexAttribPointer(shader.a_minzoom,      1, ubyte,    false, stride, 9);\n    gl.vertexAttribPointer(shader.a_maxzoom,      1, ubyte,    false, stride, 10);\n    gl.vertexAttribPointer(shader.a_angle,        1, ubyte,    false, stride, 11);\n    gl.vertexAttribPointer(shader.a_rangeend,     1, ubyte,    false, stride, 12);\n    gl.vertexAttribPointer(shader.a_rangestart,   1, ubyte,    false, stride, 13);\n\n    if (text) {\n        gl.vertexAttribPointer(shader.a_tex,          2, ubyte,     false, stride, 14);\n    } else {\n        gl.vertexAttribPointer(shader.a_tex,          2, gl.SHORT,  false, stride, 16);\n    }\n\n    // Convert the -pi..pi to an int8 range.\n    var angle = Math.round((painter.transform.angle) / Math.PI * 128);\n\n    // adjust min/max zooms for variable font sies\n    var zoomAdjust = Math.log(fontSize / info[prefix + '-max-size']) / Math.LN2 || 0;\n\n    var flip = alignedWithMap && info[prefix + '-keep-upright'];\n    gl.uniform1f(shader.u_flip, flip ? 1 : 0);\n    gl.uniform1f(shader.u_angle, (angle + 256) % 256);\n    gl.uniform1f(shader.u_zoom, (painter.transform.zoom - zoomAdjust) * 10); // current zoom level\n\n    var f = painter.frameHistory.getFadeProperties(300);\n    gl.uniform1f(shader.u_fadedist, f.fadedist * 10);\n    gl.uniform1f(shader.u_minfadezoom, Math.floor(f.minfadezoom * 10));\n    gl.uniform1f(shader.u_maxfadezoom, Math.floor(f.maxfadezoom * 10));\n    gl.uniform1f(shader.u_fadezoom, (painter.transform.zoom + f.bump) * 10);\n\n    if (!sdf) gl.uniform1f(shader.u_opacity, layerStyle['icon-opacity']);\n\n    var sdfFontSize = text ? 24 : 1;\n    var sdfPx = 8;\n    var blurOffset = 1.19;\n    var haloOffset = 6;\n\n    if (sdf) {\n\n        gl.uniform1f(shader.u_gamma, 0.105 * sdfFontSize / fontSize / browser.devicePixelRatio);\n        gl.uniform4fv(shader.u_color, layerStyle[prefix + '-color']);\n        gl.uniform1f(shader.u_buffer, (256 - 64) / 256);\n    }\n\n    var begin = bucket.elementGroups[prefix].groups[0].vertexStartIndex,\n        len = bucket.elementGroups[prefix].groups[0].vertexLength;\n\n    gl.drawArrays(gl.TRIANGLES, begin, len);\n\n    if (sdf && layerStyle[prefix + '-halo-color']) {\n        // Draw halo underneath the text.\n        gl.uniform1f(shader.u_gamma, (layerStyle[prefix + '-halo-blur'] * blurOffset / (fontSize / sdfFontSize) / sdfPx) + (0.105 * sdfFontSize / fontSize) / browser.devicePixelRatio);\n        gl.uniform4fv(shader.u_color, layerStyle[prefix + '-halo-color']);\n        gl.uniform1f(shader.u_buffer, (haloOffset - layerStyle[prefix + '-halo-width'] / (fontSize / sdfFontSize)) / sdfPx);\n\n        gl.drawArrays(gl.TRIANGLES, begin, len);\n    }\n}\n","'use strict';\n\nvar browser = require('../util/browser.js');\nvar mat4 = require('../lib/glmatrix.js').mat4;\n\nmodule.exports = drawVertices;\n\nfunction drawVertices(gl, painter, bucket) {\n    // Blend to the front, not the back.\n    gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA);\n\n    gl.switchShader(painter.dotShader, painter.tile.posMatrix, painter.tile.exMatrix);\n\n    // // Draw debug points.\n    gl.uniform1f(painter.dotShader.u_size, 4 * browser.devicePixelRatio);\n    gl.uniform1f(painter.dotShader.u_blur, 0.25);\n    gl.uniform4fv(painter.dotShader.u_color, [0.25, 0, 0, 0.25]);\n\n    // Draw the actual triangle fan into the stencil buffer.\n\n    var vertex, groups, group, begin, count;\n\n    // Draw all buffers\n    if (bucket.info.fill) {\n        vertex = bucket.buffers.fillVertex;\n        vertex.bind(gl);\n        groups = bucket.elementGroups.groups;\n        for (var i = 0; i < groups.length; i++) {\n            group = groups[i];\n            begin = group.vertexStartIndex;\n            count = group.vertexLength;\n            gl.vertexAttribPointer(painter.dotShader.a_pos, 2, gl.SHORT, false, 0, 0);\n            gl.drawArrays(gl.POINTS, begin, count);\n        }\n    }\n\n    var newPosMatrix = mat4.clone(painter.tile.posMatrix);\n    mat4.scale(newPosMatrix, newPosMatrix, [0.5, 0.5, 1]);\n\n    gl.switchShader(painter.dotShader, newPosMatrix, painter.tile.exMatrix);\n\n    // Draw all line buffers\n    if (bucket.info.line) {\n        vertex = bucket.buffers.lineVertex;\n        vertex.bind(gl);\n        groups = bucket.elementGroups.groups;\n        for (var k = 0; k < groups.length; k++) {\n            group = groups[k];\n            begin = group.vertexStartIndex;\n            count = group.vertexLength;\n            gl.vertexAttribPointer(painter.dotShader.a_pos, 2, gl.SHORT, false, 0, 0);\n            gl.drawArrays(gl.POINTS, begin, count);\n        }\n\n    }\n\n    // Revert blending mode to blend to the back.\n    gl.blendFunc(gl.ONE_MINUS_DST_ALPHA, gl.ONE);\n}\n","'use strict';\n\nmodule.exports = FrameHistory;\n\nfunction FrameHistory() {\n}\n\nFrameHistory.prototype.getFadeProperties = function(duration) {\n    if (duration === undefined) duration = 300;\n    var currentTime = (new Date()).getTime();\n\n    // Remove frames until only one is outside the duration, or until there are only three\n    while (frameHistory.length > 3 && frameHistory[1].time + duration < currentTime) {\n        frameHistory.shift();\n    }\n\n    if (frameHistory[1].time + duration < currentTime) {\n        frameHistory[0].z = frameHistory[1].z;\n    }\n\n    var frameLen = frameHistory.length;\n    if (frameLen < 3) console.warn('there should never be less than three frames in the history');\n\n    // Find the range of zoom levels we want to fade between\n    var startingZ = frameHistory[0].z,\n        lastFrame = frameHistory[frameLen - 1],\n        endingZ = lastFrame.z,\n        lowZ = Math.min(startingZ, endingZ),\n        highZ = Math.max(startingZ, endingZ);\n\n    // Calculate the speed of zooming, and how far it would zoom in terms of zoom levels in one duration\n    var zoomDiff = lastFrame.z - frameHistory[1].z,\n        timeDiff = lastFrame.time - frameHistory[1].time;\n    var fadedist = zoomDiff / (timeDiff / duration);\n\n    if (isNaN(fadedist)) console.warn('fadedist should never be NaN');\n\n    // At end of a zoom when the zoom stops changing continue pretending to zoom at that speed\n    // bump is how much farther it would have been if it had continued zooming at the same rate\n    var bump = (currentTime - lastFrame.time) / duration * fadedist;\n\n    return {\n        fadedist: fadedist,\n        minfadezoom: lowZ,\n        maxfadezoom: highZ,\n        bump: bump\n    };\n};\n\n// Store previous render times\nvar frameHistory = [];\n\n// Record frame history that will be used to calculate fading params\nFrameHistory.prototype.record = function(zoom) {\n    var currentTime = (new Date()).getTime();\n\n    // first frame ever\n    if (!frameHistory.length) {\n        frameHistory.push({time: 0, z: zoom }, {time: 0, z: zoom });\n    }\n\n    if (frameHistory.length === 2 || frameHistory[frameHistory.length - 1].z !== zoom) {\n        frameHistory.push({\n            time: currentTime,\n            z: zoom\n        });\n    }\n};\n\n","'use strict';\n\nvar shaders = require('./shaders.js');\n\nexports.extend = function(context) {\n    context.getShader = function(name, type) {\n        var kind = type == this.FRAGMENT_SHADER ? 'fragment' : 'vertex';\n        if (!shaders[name] || !shaders[name][kind]) {\n            throw new Error(\"Could not find shader \" + name);\n        }\n\n        var shader = this.createShader(type);\n        this.shaderSource(shader, shaders[name][kind]);\n        this.compileShader(shader);\n        if (!this.getShaderParameter(shader, this.COMPILE_STATUS)) {\n            throw new Error(this.getShaderInfoLog(shader));\n        }\n        return shader;\n    };\n\n    context.initializeShader = function(name, attributes, uniforms) {\n        var shader = {\n            program: this.createProgram(),\n            fragment: this.getShader(name, this.FRAGMENT_SHADER),\n            vertex: this.getShader(name, this.VERTEX_SHADER),\n            attributes: []\n        };\n        this.attachShader(shader.program, shader.vertex);\n        this.attachShader(shader.program, shader.fragment);\n        this.linkProgram(shader.program);\n\n        if (!this.getProgramParameter(shader.program, this.LINK_STATUS)) {\n            console.error(this.getProgramInfoLog(shader.program));\n            alert(\"Could not initialize shader \" + name);\n        } else {\n            for (var i = 0; i < attributes.length; i++) {\n                shader[attributes[i]] = this.getAttribLocation(shader.program, attributes[i]);\n                shader.attributes.push(shader[attributes[i]]);\n            }\n            for (var k = 0; k < uniforms.length; k++) {\n                shader[uniforms[k]] = this.getUniformLocation(shader.program, uniforms[k]);\n            }\n        }\n\n        return shader;\n    };\n\n    // Switches to a different shader program.\n    context.switchShader = function(shader, posMatrix, exMatrix) {\n        if (!posMatrix) {\n            console.trace('posMatrix does not have required argument');\n        }\n\n        if (this.currentShader !== shader) {\n            this.useProgram(shader.program);\n\n            // Disable all attributes from the existing shader that aren't used in\n            // the new shader. Note: attribute indices are *not* program specific!\n            var enabled = this.currentShader ? this.currentShader.attributes : [];\n            var required = shader.attributes;\n\n            for (var i = 0; i < enabled.length; i++) {\n                if (required.indexOf(enabled[i]) < 0) {\n                    this.disableVertexAttribArray(enabled[i]);\n                }\n            }\n\n            // Enable all attributes for the new shader.\n            for (var j = 0; j < required.length; j++) {\n                if (enabled.indexOf(required[j]) < 0) {\n                    this.enableVertexAttribArray(required[j]);\n                }\n            }\n\n            this.currentShader = shader;\n        }\n\n        // Update the matrices if necessary. Note: This relies on object identity!\n        // This means changing the matrix values without the actual matrix object\n        // will FAIL to update the matrix properly.\n        if (shader.posMatrix !== posMatrix) {\n            this.uniformMatrix4fv(shader.u_posmatrix, false, posMatrix);\n            shader.posMatrix = posMatrix;\n        }\n        if (exMatrix && shader.exMatrix !== exMatrix && shader.u_exmatrix) {\n            this.uniformMatrix4fv(shader.u_exmatrix, false, exMatrix);\n            shader.exMatrix = exMatrix;\n        }\n    };\n\n    return context;\n};\n","'use strict';\n\nvar glutil = require('./glutil.js');\nvar browser = require('../util/browser.js');\nvar GlyphAtlas = require('../symbol/glyphatlas.js');\nvar glmatrix = require('../lib/glmatrix.js');\nvar FrameHistory = require('./framehistory.js');\n\nvar mat4 = glmatrix.mat4;\n\nvar drawSymbol = require('./drawsymbol.js');\nvar drawLine = require('./drawline.js');\nvar drawFill = require('./drawfill.js');\nvar drawRaster = require('./drawraster.js');\nvar drawDebug = require('./drawdebug.js');\nvar drawBackground = require('./drawbackground.js');\nvar drawComposited = require('./drawcomposited.js');\nvar drawVertices = require('./drawvertices.js');\n\n/*\n * Initialize a new painter object.\n *\n * @param {Canvas} gl an experimental-webgl drawing context\n */\nmodule.exports = GLPainter;\nfunction GLPainter(gl, transform) {\n    this.gl = glutil.extend(gl);\n    this.transform = transform;\n\n    this.framebufferObject = null;\n    this.renderTextures = [];\n    this.namedRenderTextures = {};\n    this.reusableTextures = {};\n    this.preFbos = {};\n\n    this.tileExtent = 4096;\n    this.frameHistory = new FrameHistory();\n\n    this.setup();\n}\n\n/*\n * Update the GL viewport, projection matrix, and transforms to compensate\n * for a new width and height value.\n */\nGLPainter.prototype.resize = function(width, height) {\n    var gl = this.gl;\n    // Initialize projection matrix\n    this.projectionMatrix = mat4.create();\n    mat4.ortho(this.projectionMatrix, 0, width, height, 0, 0, -1);\n\n    this.width = width * browser.devicePixelRatio;\n    this.height = height * browser.devicePixelRatio;\n    gl.viewport(0, 0, this.width, this.height);\n\n    for (var i = this.renderTextures.length - 1; i >= 0; i--) {\n        gl.deleteTexture(this.renderTextures.pop());\n    }\n    if (this.stencilBuffer) {\n        gl.deleteRenderbuffer(this.stencilBuffer);\n        delete this.stencilBuffer;\n    }\n};\n\n\nGLPainter.prototype.setup = function() {\n    var gl = this.gl;\n\n    gl.verbose = true;\n\n    // We are blending the new pixels *behind* the existing pixels. That way we can\n    // draw front-to-back and use then stencil buffer to cull opaque pixels early.\n    gl.enable(gl.BLEND);\n    gl.blendFunc(gl.ONE_MINUS_DST_ALPHA, gl.ONE);\n\n    gl.enable(gl.STENCIL_TEST);\n\n    this.glyphAtlas = new GlyphAtlas(1024, 1024);\n    // this.glyphAtlas.debug = true;\n    this.glyphAtlas.bind(gl);\n\n    // Initialize shaders\n    this.debugShader = gl.initializeShader('debug',\n        ['a_pos'],\n        ['u_posmatrix', 'u_pointsize', 'u_color']);\n\n    this.compositeShader = gl.initializeShader('composite',\n        ['a_pos'],\n        ['u_posmatrix', 'u_opacity']);\n\n    this.gaussianShader = gl.initializeShader('gaussian',\n        ['a_pos'],\n        ['u_posmatrix', 'u_opacity', 'u_image', 'u_offset']);\n\n    this.rasterShader = gl.initializeShader('raster',\n        ['a_pos', 'a_texture_pos'],\n        ['u_posmatrix', 'u_brightness_low', 'u_brightness_high', 'u_saturation_factor', 'u_spin_weights', 'u_contrast_factor', 'u_opacity0', 'u_opacity1', 'u_image0', 'u_image1', 'u_tl_parent', 'u_scale_parent', 'u_buffer_scale']);\n\n    this.lineShader = gl.initializeShader('line',\n        ['a_pos', 'a_extrude', 'a_linesofar'],\n        ['u_posmatrix', 'u_exmatrix', 'u_linewidth', 'u_color', 'u_debug', 'u_ratio', 'u_dasharray', 'u_blur']);\n\n    this.linepatternShader = gl.initializeShader('linepattern',\n        ['a_pos', 'a_extrude', 'a_linesofar'],\n        ['u_posmatrix', 'u_exmatrix', 'u_linewidth', 'u_ratio', 'u_pattern_size', 'u_pattern_tl', 'u_pattern_br', 'u_point', 'u_blur', 'u_fade']);\n\n    this.dotShader = gl.initializeShader('dot',\n        ['a_pos'],\n        ['u_posmatrix', 'u_size', 'u_color', 'u_blur']);\n\n    this.sdfShader = gl.initializeShader('sdf',\n        ['a_pos', 'a_tex', 'a_offset', 'a_angle', 'a_minzoom', 'a_maxzoom', 'a_rangeend', 'a_rangestart', 'a_labelminzoom'],\n        ['u_posmatrix', 'u_exmatrix', 'u_texture', 'u_texsize', 'u_color', 'u_gamma', 'u_buffer', 'u_angle', 'u_zoom', 'u_flip', 'u_fadedist', 'u_minfadezoom', 'u_maxfadezoom', 'u_fadezoom']);\n\n    this.iconShader = gl.initializeShader('icon',\n        ['a_pos', 'a_tex', 'a_offset', 'a_angle', 'a_minzoom', 'a_maxzoom', 'a_rangeend', 'a_rangestart', 'a_labelminzoom'],\n        ['u_posmatrix', 'u_exmatrix', 'u_texture', 'u_texsize', 'u_angle', 'u_zoom', 'u_flip', 'u_fadedist', 'u_minfadezoom', 'u_maxfadezoom', 'u_fadezoom', 'u_opacity']);\n\n    this.outlineShader = gl.initializeShader('outline',\n        ['a_pos'],\n        ['u_posmatrix', 'u_color', 'u_world']\n    );\n\n    this.patternShader = gl.initializeShader('pattern',\n        ['a_pos'],\n        ['u_posmatrix', 'u_pattern_tl', 'u_pattern_br', 'u_mix', 'u_patternmatrix', 'u_opacity', 'u_image']\n    );\n\n    this.fillShader = gl.initializeShader('fill',\n        ['a_pos'],\n        ['u_posmatrix', 'u_color']\n    );\n\n    this.identityMatrix = mat4.create();\n\n    // The backgroundBuffer is used when drawing to the full *canvas*\n    this.backgroundBuffer = gl.createBuffer();\n    this.backgroundBuffer.itemSize = 2;\n    this.backgroundBuffer.itemCount = 4;\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.backgroundBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, new Int16Array([-1, -1, 1, -1, -1, 1, 1, 1]), gl.STATIC_DRAW);\n\n    // The tileExtentBuffer is used when drawing to a full *tile*\n    this.tileExtentBuffer = gl.createBuffer();\n    this.tileExtentBuffer.itemSize = 4;\n    this.tileExtentBuffer.itemCount = 4;\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.tileExtentBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, new Int16Array([\n        // tile coord x, tile coord y, texture coord x, texture coord y\n                      0, 0,                    0, 0,\n        this.tileExtent, 0,                32767, 0,\n                      0, this.tileExtent,      0, 32767,\n        this.tileExtent, this.tileExtent,  32767, 32767\n    ]), gl.STATIC_DRAW);\n\n    // The debugBuffer is used to draw tile outlines for debugging\n    this.debugBuffer = gl.createBuffer();\n    this.debugBuffer.itemSize = 2;\n    this.debugBuffer.itemCount = 5;\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.debugBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, new Int16Array([0, 0, 4095, 0, 4095, 4095, 0, 4095, 0, 0]), gl.STATIC_DRAW);\n\n    // The debugTextBuffer is used to draw tile IDs for debugging\n    this.debugTextBuffer = gl.createBuffer();\n    this.debugTextBuffer.itemSize = 2;\n};\n\n/*\n * Reset the color buffers of the drawing canvas.\n */\nGLPainter.prototype.clearColor = function() {\n    var gl = this.gl;\n    gl.clearColor(0, 0, 0, 0);\n    gl.clear(gl.COLOR_BUFFER_BIT);\n};\n\n/*\n * Reset the drawing canvas by clearing the stencil buffer so that we can draw\n * new tiles at the same location, while retaining previously drawn pixels.\n */\nGLPainter.prototype.clearStencil = function() {\n    var gl = this.gl;\n    gl.clearStencil(0x0);\n    gl.stencilMask(0xFF);\n    gl.clear(gl.STENCIL_BUFFER_BIT);\n};\n\nGLPainter.prototype.drawClippingMask = function() {\n    var gl = this.gl;\n    gl.switchShader(this.fillShader, this.tile.posMatrix, this.tile.exMatrix);\n    gl.colorMask(false, false, false, false);\n\n    // Clear the entire stencil buffer, except for the 7th bit, which stores\n    // the global clipping mask that allows us to avoid drawing in regions of\n    // tiles we've already painted in.\n    gl.clearStencil(0x0);\n    gl.stencilMask(0xBF);\n    gl.clear(gl.STENCIL_BUFFER_BIT);\n\n    // The stencil test will fail always, meaning we set all pixels covered\n    // by this geometry to 0x80. We use the highest bit 0x80 to mark the regions\n    // we want to draw in. All pixels that have this bit *not* set will never be\n    // drawn in.\n    gl.stencilFunc(gl.EQUAL, 0xC0, 0x40);\n    gl.stencilMask(0xC0);\n    gl.stencilOp(gl.REPLACE, gl.KEEP, gl.KEEP);\n\n    // Draw the clipping mask\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.tileExtentBuffer);\n    gl.vertexAttribPointer(this.fillShader.a_pos, this.tileExtentBuffer.itemSize, gl.SHORT, false, 8, 0);\n    gl.drawArrays(gl.TRIANGLE_STRIP, 0, this.tileExtentBuffer.itemCount);\n\n    gl.stencilFunc(gl.EQUAL, 0x80, 0x80);\n    gl.stencilOp(gl.KEEP, gl.KEEP, gl.REPLACE);\n    gl.stencilMask(0x00);\n    gl.colorMask(true, true, true, true);\n};\n\n// Set up a texture that can be drawn into\nGLPainter.prototype.bindRenderTexture = function(name) {\n    var gl = this.gl;\n    if (name) {\n\n        // Only create one framebuffer. Reuse it for every level.\n        if (!this.framebufferObject) {\n            this.framebufferObject = gl.createFramebuffer();\n        }\n\n        gl.bindFramebuffer(gl.FRAMEBUFFER, this.framebufferObject);\n\n        // There's only one stencil buffer that we always attach.\n        if (!this.stencilBuffer) {\n            var stencil = this.stencilBuffer = gl.createRenderbuffer();\n            gl.bindRenderbuffer(gl.RENDERBUFFER, stencil);\n            gl.renderbufferStorage(gl.RENDERBUFFER, gl.STENCIL_INDEX8, gl.drawingBufferWidth, gl.drawingBufferHeight);\n            gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.STENCIL_ATTACHMENT, gl.RENDERBUFFER, this.stencilBuffer);\n        }\n\n        // We create a separate texture for every level.\n        var texture = this.renderTextures.pop();\n        if (!texture) {\n            texture = gl.createTexture();\n            gl.bindTexture(gl.TEXTURE_2D, texture);\n            gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n            gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n            gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST);\n            gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);\n            gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.drawingBufferWidth, gl.drawingBufferHeight, 0, gl.RGBA, gl.UNSIGNED_BYTE, null);\n        }\n\n        this.namedRenderTextures[name] = texture;\n\n        gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, texture, 0);\n\n    } else {\n        gl.bindFramebuffer(gl.FRAMEBUFFER, null);\n    }\n\n    this.clearColor();\n};\n\nGLPainter.prototype.freeRenderTexture = function(name) {\n    this.renderTextures.push(this.namedRenderTextures[name]);\n    delete this.namedRenderTextures[name];\n};\n\n/*\n * Draw a new tile to the context, assuming that the viewport is\n * already correctly set.\n */\nGLPainter.prototype.draw = function glPainterDraw(tile, style, layers, params, matrix) {\n    this.tile = tile;\n\n    // false when drawing a group of composited layers\n    if (tile && !matrix) {\n        // Draw the root clipping mask.\n        this.drawClippingMask();\n    }\n\n    if (!Array.isArray(layers)) console.warn('Layers is not an array');\n\n    this.frameHistory.record(this.transform.zoom);\n\n    // Draw layers front-to-back.\n    // Layers are already in reverse order from style.restructure()\n    for (var i = 0, len = layers.length; i < len; i++) {\n        this.applyStyle(layers[i], style, tile && tile.buckets, params, tile, matrix);\n    }\n\n    if (params.debug) {\n        drawDebug(this.gl, this, tile, params);\n    }\n};\n\nGLPainter.prototype.applyStyle = function(layer, style, buckets, params, tile, matrix) {\n    var gl = this.gl;\n\n    var layerStyle = style.computed[layer.id];\n    if (!layerStyle || layerStyle.hidden) return;\n\n    if (layer.layers) {\n        if (layer.type === 'composite') {\n            drawComposited(gl, this, buckets, layerStyle, params, style, layer);\n        } else if (layer.type === 'raster') {\n            drawRaster(gl, this, buckets[layer.bucket], layerStyle, params, style, layer, tile);\n        }\n    } else if (params.background) {\n        drawBackground(gl, this, undefined, layerStyle, this.identityMatrix, params, style.sprite);\n    } else {\n\n        var bucket = buckets[layer.bucket];\n        // There are no vertices yet for this layer.\n        if (!bucket || (bucket.hasData && !bucket.hasData())) return;\n\n        var type = bucket.type;\n\n        if (bucket.minZoom && this.transform.zoom < bucket.minZoom) return;\n        if (bucket.maxZoom && this.transform.zoom >= bucket.maxZoom) return;\n\n        var draw = type === 'symbol' ? drawSymbol :\n                   type === 'fill' ? drawFill :\n                   type === 'line' ? drawLine :\n                   type === 'raster' ? drawRaster : null;\n\n        if (draw) {\n            var useMatrix = matrix || this.tile.posMatrix;\n            draw(gl, this, bucket, layerStyle, useMatrix, params, style.sprite);\n        } else {\n            console.warn('No bucket type specified');\n        }\n\n        if (params.vertices && !layer.layers) {\n            drawVertices(gl, this, bucket);\n        }\n    }\n};\n\n// Draws non-opaque areas. This is for debugging purposes.\nGLPainter.prototype.drawStencilBuffer = function() {\n    var gl = this.gl;\n    gl.switchShader(this.fillShader, this.identityMatrix);\n\n    // Blend to the front, not the back.\n    gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA);\n    gl.stencilMask(0x00);\n    gl.stencilFunc(gl.EQUAL, 0x80, 0x80);\n\n    // Drw the filling quad where the stencil buffer isn't set.\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.backgroundBuffer);\n    gl.vertexAttribPointer(this.fillShader.a_pos, this.backgroundBuffer.itemSize, gl.SHORT, false, 0, 0);\n    gl.uniform4fv(this.fillShader.u_color, [0, 0, 0, 0.5]);\n    gl.drawArrays(gl.TRIANGLE_STRIP, 0, this.backgroundBuffer.itemCount);\n\n    // Revert blending mode to blend to the back.\n    gl.blendFunc(gl.ONE_MINUS_DST_ALPHA, gl.ONE);\n};\n\nGLPainter.prototype.translateMatrix = function(matrix, z, translate, anchor) {\n    if (!translate[0] && !translate[1]) return matrix;\n\n    if (anchor === 'viewport') {\n        var sin_a = Math.sin(-this.transform.angle);\n        var cos_a = Math.cos(-this.transform.angle);\n        translate = [\n            translate[0] * cos_a - translate[1] * sin_a,\n            translate[0] * sin_a + translate[1] * cos_a\n        ];\n    }\n\n    var tilePixelRatio = this.transform.scale / (1 << z) / 8;\n    var translation = [\n        translate[0] / tilePixelRatio,\n        translate[1] / tilePixelRatio,\n        0\n    ];\n\n    var translatedMatrix = new Float32Array(16);\n    mat4.translate(translatedMatrix, matrix, translation);\n    return translatedMatrix;\n};\n\nGLPainter.prototype.saveTexture = function(texture) {\n    var textures = this.reusableTextures[texture.size];\n    if (!textures) {\n        this.reusableTextures[texture.size] = [texture];\n    } else {\n        textures.push(texture);\n    }\n};\n\n\nGLPainter.prototype.getTexture = function(size) {\n    var textures = this.reusableTextures[size];\n    return textures && textures.length > 0 ? textures.pop() : null;\n};\n","'use strict';\n\nvar glmatrix = require('../lib/glmatrix.js');\nvar mat4 = glmatrix.mat4;\n\nmodule.exports = PrerenderedTexture;\n\nfunction PrerenderedTexture(gl, bucket, painter) {\n    this.gl = gl;\n    this.buffer = bucket['raster-buffer'] || (1/32);\n    this.size = (bucket['raster-size'] || 512) * (1 + 2 * this.buffer);\n    this.painter = painter;\n\n    this.texture = null;\n    this.fbo = null;\n    this.fbos = this.painter.preFbos[this.size];\n}\n\nPrerenderedTexture.prototype.bindFramebuffer = function() {\n    var gl = this.gl;\n\n    // try to reuse available raster textures\n    this.texture = this.painter.getTexture(this.size);\n\n    if (!this.texture) {\n        this.texture = gl.createTexture();\n        gl.bindTexture(gl.TEXTURE_2D, this.texture);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR);\n        gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, this.size, this.size, 0, gl.RGBA, gl.UNSIGNED_BYTE, null);\n        this.texture.size = this.size;\n    } else {\n        gl.bindTexture(gl.TEXTURE_2D, this.texture);\n    }\n\n    if (!this.fbos) {\n        this.fbo = gl.createFramebuffer();\n        var stencil = gl.createRenderbuffer();\n        gl.bindRenderbuffer(gl.RENDERBUFFER, stencil);\n        gl.renderbufferStorage(gl.RENDERBUFFER, gl.STENCIL_INDEX8, this.size, this.size);\n        gl.bindFramebuffer(gl.FRAMEBUFFER, this.fbo);\n        gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.STENCIL_ATTACHMENT, gl.RENDERBUFFER, stencil);\n        gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, this.texture, 0);\n    } else {\n        this.fbo = this.fbos.pop();\n        gl.bindFramebuffer(gl.FRAMEBUFFER, this.fbo);\n        gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, this.texture, 0);\n    }\n};\n\nPrerenderedTexture.prototype.unbindFramebuffer = function() {\n    var gl = this.gl;\n    gl.bindFramebuffer(gl.FRAMEBUFFER, null);\n    if (this.fbos) this.fbos.push(this.fbo); else this.painter.preFbos[this.size] = [this.fbo];\n};\n\nPrerenderedTexture.prototype.bind = function() {\n    if (!this.texture) throw('pre-rendered texture does not exist');\n    var gl = this.gl;\n    gl.bindTexture(gl.TEXTURE_2D, this.texture);\n};\n\nPrerenderedTexture.prototype.blur = function(painter, passes) {\n    var gl = this.gl;\n    var originalTexture = this.texture;\n    var secondaryTexture = this.painter.getTexture(this.size);\n    if (!secondaryTexture) {\n        secondaryTexture = gl.createTexture();\n        gl.bindTexture(gl.TEXTURE_2D, secondaryTexture);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR);\n        gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, this.size, this.size, 0, gl.RGBA, gl.UNSIGNED_BYTE, null);\n        secondaryTexture.size = this.size;\n    } else {\n        gl.bindTexture(gl.TEXTURE_2D, secondaryTexture);\n    }\n    gl.bindTexture(gl.TEXTURE_2D, null);\n\n    var matrix = mat4.create();\n    mat4.ortho(matrix, 0, 4096, -4096, 0, 0, 1);\n    mat4.translate(matrix, matrix, [0, -4096, 0]);\n\n    gl.switchShader(painter.gaussianShader, matrix);\n    gl.activeTexture(gl.TEXTURE0);\n    gl.uniform1i(painter.gaussianShader.u_image, 0);\n    gl.uniform1f(painter.gaussianShader.u_opacity, 1);\n\n    for (var i = 0; i < passes; i++) {\n\n        // Render horizontal\n        gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, secondaryTexture, 0);\n        gl.clear(gl.COLOR_BUFFER_BIT);\n        gl.uniform2fv(painter.gaussianShader.u_offset, [1 / this.size, 0]);\n        gl.bindTexture(gl.TEXTURE_2D, originalTexture);\n        gl.bindBuffer(gl.ARRAY_BUFFER, painter.tileExtentBuffer);\n        gl.vertexAttribPointer(painter.gaussianShader.a_pos, 2, gl.SHORT, false, 8, 0);\n        gl.drawArrays(gl.TRIANGLE_STRIP, 0, 4);\n\n\n        // Render vertical\n        gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, originalTexture, 0);\n        gl.clear(gl.COLOR_BUFFER_BIT);\n        gl.uniform2fv(painter.gaussianShader.u_offset, [0, 1 / this.size]);\n        gl.bindTexture(gl.TEXTURE_2D, secondaryTexture);\n        gl.bindBuffer(gl.ARRAY_BUFFER, painter.tileExtentBuffer);\n        gl.vertexAttribPointer(painter.gaussianShader.a_pos, 2, gl.SHORT, false, 8, 0);\n        gl.drawArrays(gl.TRIANGLE_STRIP, 0, 4);\n    }\n\n    this.painter.saveTexture(secondaryTexture);\n};\n","'use strict';\n\nvar glify = undefined;\n\nmodule.exports = {\n    \"composite\": {\"vertex\":\"precision mediump float;\\nattribute vec2 a_pos;\\nuniform mat4 u_posmatrix;\\nvarying highp vec2 a;\\nvoid main ()\\n{\\n  vec4 tmpvar_1;\\n  tmpvar_1.zw = vec2(0.0, 1.0);\\n  tmpvar_1.xy = a_pos;\\n  vec4 tmpvar_2;\\n  tmpvar_2 = (u_posmatrix * tmpvar_1);\\n  gl_Position = tmpvar_2;\\n  a = ((tmpvar_2.xy / 2.0) + 0.5);\\n}\\n\\n\",\"fragment\":\"precision mediump float;\\nuniform sampler2D u_image;\\nuniform float u_opacity;\\nvarying vec2 a;\\nvoid main ()\\n{\\n  lowp vec4 tmpvar_1;\\n  tmpvar_1 = (texture2D (u_image, a) * u_opacity);\\n  gl_FragColor = tmpvar_1;\\n}\\n\\n\"},\n    \"debug\": {\"vertex\":\"precision mediump float;\\nattribute vec2 a_pos;\\nuniform float u_pointsize;\\nuniform mat4 u_posmatrix;\\nvoid main ()\\n{\\n  vec4 tmpvar_1;\\n  tmpvar_1.w = 1.0;\\n  tmpvar_1.xy = a_pos;\\n  tmpvar_1.z = float((a_pos.x >= 32767.0));\\n  gl_Position = (u_posmatrix * tmpvar_1);\\n  gl_PointSize = u_pointsize;\\n}\\n\\n\",\"fragment\":\"precision mediump float;\\nuniform vec4 u_color;\\nvoid main ()\\n{\\n  gl_FragColor = u_color;\\n}\\n\\n\"},\n    \"dot\": {\"vertex\":\"precision mediump float;\\nuniform mat4 u_posmatrix;\\nuniform float u_size;\\nattribute vec2 a_pos;\\nvoid main ()\\n{\\n  vec4 tmpvar_1;\\n  tmpvar_1.zw = vec2(0.0, 1.0);\\n  tmpvar_1.xy = a_pos;\\n  gl_Position = (u_posmatrix * tmpvar_1);\\n  gl_PointSize = u_size;\\n}\\n\\n\",\"fragment\":\"precision mediump float;\\nuniform vec4 u_color;\\nuniform float u_blur;\\nvoid main ()\\n{\\n  mediump vec2 x_1;\\n  x_1 = (gl_PointCoord - 0.5);\\n  mediump float tmpvar_2;\\n  tmpvar_2 = clamp (((\\n    sqrt(dot (x_1, x_1))\\n   - 0.5) / (\\n    (0.5 - u_blur)\\n   - 0.5)), 0.0, 1.0);\\n  gl_FragColor = (u_color * (tmpvar_2 * (tmpvar_2 * \\n    (3.0 - (2.0 * tmpvar_2))\\n  )));\\n}\\n\\n\"},\n    \"fill\": {\"vertex\":\"precision mediump float;\\nattribute vec2 a_pos;\\nuniform mat4 u_posmatrix;\\nvoid main ()\\n{\\n  vec4 tmpvar_1;\\n  tmpvar_1.zw = vec2(0.0, 1.0);\\n  tmpvar_1.xy = a_pos;\\n  gl_Position = (u_posmatrix * tmpvar_1);\\n  gl_PointSize = 2.0;\\n}\\n\\n\",\"fragment\":\"precision mediump float;\\nuniform vec4 u_color;\\nvoid main ()\\n{\\n  gl_FragColor = u_color;\\n}\\n\\n\"},\n    \"gaussian\": {\"vertex\":\"precision mediump float;\\nattribute vec2 a_pos;\\nuniform mat4 u_posmatrix;\\nuniform vec2 u_offset;\\nvarying highp vec2 b[3];\\nvoid main ()\\n{\\n  vec4 tmpvar_1;\\n  tmpvar_1.zw = vec2(0.0, 1.0);\\n  tmpvar_1.xy = a_pos;\\n  vec4 tmpvar_2;\\n  tmpvar_2 = (u_posmatrix * tmpvar_1);\\n  gl_Position = tmpvar_2;\\n  highp vec2 tmpvar_3;\\n  tmpvar_3 = ((tmpvar_2.xy / 2.0) + 0.5);\\n  b[0] = tmpvar_3;\\n  vec2 cse_4;\\n  cse_4 = (u_offset * 1.18243);\\n  b[1] = (tmpvar_3 + cse_4);\\n  b[2] = (tmpvar_3 - cse_4);\\n}\\n\\n\",\"fragment\":\"precision mediump float;\\nuniform sampler2D u_image;\\nvarying vec2 b[3];\\nvoid main ()\\n{\\n  lowp vec4 tmpvar_1;\\n  tmpvar_1 = (((texture2D (u_image, b[0]) * 0.40262) + (texture2D (u_image, b[1]) * 0.29869)) + (texture2D (u_image, b[2]) * 0.29869));\\n  gl_FragColor = tmpvar_1;\\n}\\n\\n\"},\n    \"line\": {\"vertex\":\"precision mediump float;\\nattribute vec2 a_pos;\\nattribute vec2 a_extrude;\\nattribute float a_linesofar;\\nuniform mat4 u_posmatrix;\\nuniform mat4 u_exmatrix;\\nuniform float u_ratio;\\nuniform vec2 u_linewidth;\\nvarying vec2 a;\\nvarying float b;\\nvoid main ()\\n{\\n  vec2 c_1;\\n  vec2 tmpvar_2;\\n  tmpvar_2 = (vec2(mod (a_pos, 2.0)));\\n  c_1.x = tmpvar_2.x;\\n  c_1.y = sign((tmpvar_2.y - 0.5));\\n  a = c_1;\\n  vec4 tmpvar_3;\\n  tmpvar_3.zw = vec2(0.0, 1.0);\\n  tmpvar_3.xy = floor((a_pos / 2.0));\\n  vec4 tmpvar_4;\\n  tmpvar_4.zw = vec2(0.0, 0.0);\\n  tmpvar_4.xy = (u_linewidth.x * (a_extrude / 63.0));\\n  gl_Position = ((u_posmatrix * tmpvar_3) + (u_exmatrix * tmpvar_4));\\n  b = (a_linesofar * u_ratio);\\n}\\n\\n\",\"fragment\":\"precision mediump float;\\nuniform float u_debug;\\nuniform float u_blur;\\nuniform vec2 u_linewidth;\\nuniform vec2 u_dasharray;\\nuniform vec4 u_color;\\nvarying vec2 a;\\nvarying float b;\\nvoid main ()\\n{\\n  float tmpvar_1;\\n  tmpvar_1 = (sqrt(dot (a, a)) * u_linewidth.x);\\n  float tmpvar_2;\\n  tmpvar_2 = (float(mod (b, (u_dasharray.x + u_dasharray.y))));\\n  gl_FragColor = (u_color * (clamp (\\n    (min ((tmpvar_1 - (u_linewidth.y - u_blur)), (u_linewidth.x - tmpvar_1)) / u_blur)\\n  , 0.0, 1.0) * max (\\n    float((-(u_dasharray.y) >= 0.0))\\n  , \\n    clamp (min (tmpvar_2, (u_dasharray.x - tmpvar_2)), 0.0, 1.0)\\n  )));\\n  if ((u_debug > 0.0)) {\\n    gl_FragColor = vec4(1.0, 0.0, 0.0, 1.0);\\n  };\\n}\\n\\n\"},\n    \"linepattern\": {\"vertex\":\"precision mediump float;\\nattribute vec2 a_pos;\\nattribute vec2 a_extrude;\\nattribute float a_linesofar;\\nuniform mat4 u_posmatrix;\\nuniform mat4 u_exmatrix;\\nuniform float u_point;\\nuniform vec2 u_linewidth;\\nvarying vec2 a;\\nvarying float b;\\nvoid main ()\\n{\\n  vec2 c_1;\\n  vec2 tmpvar_2;\\n  tmpvar_2 = (vec2(mod (a_pos, 2.0)));\\n  c_1.x = tmpvar_2.x;\\n  c_1.y = sign((tmpvar_2.y - 0.5));\\n  a = c_1;\\n  vec4 tmpvar_3;\\n  tmpvar_3.zw = vec2(0.0, 1.0);\\n  tmpvar_3.xy = floor((a_pos / 2.0));\\n  vec4 tmpvar_4;\\n  tmpvar_4.w = 0.0;\\n  tmpvar_4.xy = ((u_linewidth.x * (a_extrude / 63.0)) * (1.0 - u_point));\\n  tmpvar_4.z = (float((a_pos.x >= 32767.0)) + (u_point * float(\\n    (c_1.y >= 1.0)\\n  )));\\n  gl_Position = ((u_posmatrix * tmpvar_3) + (u_exmatrix * tmpvar_4));\\n  b = a_linesofar;\\n  gl_PointSize = ((2.0 * u_linewidth.x) - 1.0);\\n}\\n\\n\",\"fragment\":\"precision mediump float;\\nuniform float u_point;\\nuniform float u_blur;\\nuniform float u_fade;\\nuniform vec2 u_linewidth;\\nuniform vec2 u_pattern_size;\\nuniform vec2 u_pattern_tl;\\nuniform vec2 u_pattern_br;\\nuniform sampler2D u_image;\\nvarying vec2 a;\\nvarying float b;\\nvoid main ()\\n{\\n  lowp vec4 j_1;\\n  mediump vec2 x_2;\\n  x_2 = ((gl_PointCoord * 2.0) - 1.0);\\n  mediump float tmpvar_3;\\n  tmpvar_3 = (((\\n    sqrt(dot (a, a))\\n   * \\n    (1.0 - u_point)\\n  ) + (u_point * \\n    sqrt(dot (x_2, x_2))\\n  )) * u_linewidth.x);\\n  float tmpvar_4;\\n  tmpvar_4 = (float(mod ((b / u_pattern_size.x), 1.0)));\\n  float tmpvar_5;\\n  tmpvar_5 = (0.5 + ((a.y * u_linewidth.x) / u_pattern_size.y));\\n  vec2 tmpvar_6;\\n  tmpvar_6.x = tmpvar_4;\\n  tmpvar_6.y = tmpvar_5;\\n  vec2 tmpvar_7;\\n  tmpvar_7.x = (float(mod ((tmpvar_4 * 2.0), 1.0)));\\n  tmpvar_7.y = tmpvar_5;\\n  lowp vec4 tmpvar_8;\\n  tmpvar_8 = ((texture2D (u_image, mix (u_pattern_tl, u_pattern_br, tmpvar_6)) * (1.0 - u_fade)) + (u_fade * texture2D (u_image, mix (u_pattern_tl, u_pattern_br, tmpvar_7))));\\n  j_1.w = tmpvar_8.w;\\n  j_1.xyz = (tmpvar_8.xyz * tmpvar_8.w);\\n  gl_FragColor = (j_1 * clamp ((\\n    min ((tmpvar_3 - (u_linewidth.y - u_blur)), (u_linewidth.x - tmpvar_3))\\n   / u_blur), 0.0, 1.0));\\n}\\n\\n\"},\n    \"outline\": {\"vertex\":\"precision mediump float;\\nattribute vec2 a_pos;\\nuniform mat4 u_posmatrix;\\nuniform vec2 u_world;\\nvarying highp vec2 a;\\nvoid main ()\\n{\\n  vec4 tmpvar_1;\\n  tmpvar_1.zw = vec2(0.0, 1.0);\\n  tmpvar_1.xy = a_pos;\\n  vec4 tmpvar_2;\\n  tmpvar_2 = (u_posmatrix * tmpvar_1);\\n  gl_Position = tmpvar_2;\\n  a = (((tmpvar_2.xy + 1.0) / 2.0) * u_world);\\n}\\n\\n\",\"fragment\":\"precision mediump float;\\nuniform vec4 u_color;\\nvarying vec2 a;\\nvoid main ()\\n{\\n  highp vec2 x_1;\\n  x_1 = (a - gl_FragCoord.xy);\\n  highp float tmpvar_2;\\n  tmpvar_2 = clamp (((\\n    sqrt(dot (x_1, x_1))\\n   - 1.0) / -1.0), 0.0, 1.0);\\n  highp vec4 tmpvar_3;\\n  tmpvar_3 = (u_color * (tmpvar_2 * (tmpvar_2 * \\n    (3.0 - (2.0 * tmpvar_2))\\n  )));\\n  gl_FragColor = tmpvar_3;\\n}\\n\\n\"},\n    \"pattern\": {\"vertex\":\"precision mediump float;\\nuniform mat4 u_posmatrix;\\nuniform mat3 u_patternmatrix;\\nattribute vec2 a_pos;\\nvarying vec2 a;\\nvoid main ()\\n{\\n  vec4 tmpvar_1;\\n  tmpvar_1.zw = vec2(0.0, 1.0);\\n  tmpvar_1.xy = a_pos;\\n  gl_Position = (u_posmatrix * tmpvar_1);\\n  vec3 tmpvar_2;\\n  tmpvar_2.z = 1.0;\\n  tmpvar_2.xy = a_pos;\\n  a = (u_patternmatrix * tmpvar_2).xy;\\n}\\n\\n\",\"fragment\":\"precision mediump float;\\nuniform float u_opacity;\\nuniform float u_mix;\\nuniform vec2 u_pattern_tl;\\nuniform vec2 u_pattern_br;\\nuniform sampler2D u_image;\\nvarying vec2 a;\\nvoid main ()\\n{\\n  vec2 tmpvar_1;\\n  tmpvar_1 = (vec2(mod (a, 1.0)));\\n  lowp vec4 tmpvar_2;\\n  tmpvar_2 = (mix (texture2D (u_image, mix (u_pattern_tl, u_pattern_br, tmpvar_1)), texture2D (u_image, mix (u_pattern_tl, u_pattern_br, \\n    (vec2(mod ((tmpvar_1 * 2.0), 1.0)))\\n  )), u_mix) * u_opacity);\\n  gl_FragColor = tmpvar_2;\\n}\\n\\n\"},\n    \"raster\": {\"vertex\":\"precision mediump float;\\nuniform mat4 u_posmatrix;\\nuniform vec2 u_tl_parent;\\nuniform float u_scale_parent;\\nuniform float u_buffer_scale;\\nattribute vec2 a_pos;\\nattribute vec2 a_texture_pos;\\nvarying vec2 a;\\nvarying vec2 b;\\nvoid main ()\\n{\\n  vec4 tmpvar_1;\\n  tmpvar_1.zw = vec2(0.0, 1.0);\\n  tmpvar_1.xy = a_pos;\\n  gl_Position = (u_posmatrix * tmpvar_1);\\n  vec2 tmpvar_2;\\n  tmpvar_2 = (((\\n    (a_texture_pos / 32767.0)\\n   - 0.5) / u_buffer_scale) + 0.5);\\n  a = tmpvar_2;\\n  b = ((tmpvar_2 * u_scale_parent) + u_tl_parent);\\n}\\n\\n\",\"fragment\":\"precision mediump float;\\nuniform float u_opacity0;\\nuniform float u_opacity1;\\nuniform float u_brightness_low;\\nuniform float u_brightness_high;\\nuniform float u_saturation_factor;\\nuniform float u_contrast_factor;\\nuniform sampler2D u_image0;\\nuniform sampler2D u_image1;\\nvarying vec2 a;\\nvarying vec2 b;\\nuniform vec3 u_spin_weights;\\nvoid main ()\\n{\\n  lowp vec4 tmpvar_1;\\n  tmpvar_1 = ((texture2D (u_image0, a) * u_opacity0) + (texture2D (u_image1, b) * u_opacity1));\\n  lowp vec3 tmpvar_2;\\n  tmpvar_2.x = dot (tmpvar_1.xyz, u_spin_weights);\\n  tmpvar_2.y = dot (tmpvar_1.xyz, u_spin_weights.zxy);\\n  tmpvar_2.z = dot (tmpvar_1.xyz, u_spin_weights.yzx);\\n  lowp vec4 tmpvar_3;\\n  tmpvar_3.xyz = mix (vec3(u_brightness_low), vec3(u_brightness_high), ((\\n    ((tmpvar_2 + ((\\n      (((tmpvar_1.x + tmpvar_1.y) + tmpvar_1.z) / 3.0)\\n     - tmpvar_2) * u_saturation_factor)) - 0.5)\\n   * u_contrast_factor) + 0.5));\\n  tmpvar_3.w = tmpvar_1.w;\\n  gl_FragColor = tmpvar_3;\\n}\\n\\n\"},\n    \"icon\": {\"vertex\":\"precision mediump float;\\nattribute vec2 a_pos;\\nattribute vec2 a_offset;\\nattribute vec2 a_tex;\\nattribute float a_angle;\\nattribute float a_minzoom;\\nattribute float a_maxzoom;\\nattribute float a_rangeend;\\nattribute float a_rangestart;\\nattribute float a_labelminzoom;\\nuniform mat4 u_posmatrix;\\nuniform mat4 u_exmatrix;\\nuniform float u_angle;\\nuniform float u_zoom;\\nuniform float u_flip;\\nuniform float u_fadedist;\\nuniform float u_minfadezoom;\\nuniform float u_maxfadezoom;\\nuniform float u_fadezoom;\\nuniform float u_opacity;\\nuniform vec2 u_texsize;\\nvarying vec2 a;\\nvarying float b;\\nvoid main ()\\n{\\n  float d_1;\\n  d_1 = 0.0;\\n  float tmpvar_2;\\n  tmpvar_2 = (float(mod ((a_angle + u_angle), 256.0)));\\n  if ((((u_flip > 0.0) && (tmpvar_2 >= 64.0)) && (tmpvar_2 < 192.0))) {\\n    d_1 = 1.0;\\n  };\\n  float tmpvar_3;\\n  tmpvar_3 = (((2.0 - \\n    float((u_zoom >= a_minzoom))\\n  ) - (1.0 - \\n    float((u_zoom >= a_maxzoom))\\n  )) + d_1);\\n  float tmpvar_4;\\n  tmpvar_4 = clamp (((u_fadezoom - a_labelminzoom) / u_fadedist), 0.0, 1.0);\\n  if ((u_fadedist >= 0.0)) {\\n    b = tmpvar_4;\\n  } else {\\n    b = (1.0 - tmpvar_4);\\n  };\\n  if ((u_maxfadezoom < a_labelminzoom)) {\\n    b = 0.0;\\n  };\\n  if ((u_minfadezoom >= a_labelminzoom)) {\\n    b = 1.0;\\n  };\\n  vec4 tmpvar_5;\\n  tmpvar_5.zw = vec2(0.0, 1.0);\\n  tmpvar_5.xy = a_pos;\\n  vec4 tmpvar_6;\\n  tmpvar_6.w = 0.0;\\n  tmpvar_6.xy = (a_offset / 64.0);\\n  tmpvar_6.z = ((tmpvar_3 + float(\\n    (0.0 >= b)\\n  )) + (float(\\n    (u_angle >= a_rangeend)\\n  ) * (1.0 - \\n    float((u_angle >= a_rangestart))\\n  )));\\n  gl_Position = ((u_posmatrix * tmpvar_5) + (u_exmatrix * tmpvar_6));\\n  a = (a_tex / u_texsize);\\n  b = (b * u_opacity);\\n}\\n\\n\",\"fragment\":\"precision mediump float;\\nuniform sampler2D u_texture;\\nvarying vec2 a;\\nvarying float b;\\nvoid main ()\\n{\\n  lowp vec4 tmpvar_1;\\n  tmpvar_1 = (texture2D (u_texture, a) * b);\\n  gl_FragColor = tmpvar_1;\\n}\\n\\n\"},\n    \"sdf\": {\"vertex\":\"precision mediump float;\\nattribute vec2 a_pos;\\nattribute vec2 a_offset;\\nattribute vec2 a_tex;\\nattribute float a_angle;\\nattribute float a_minzoom;\\nattribute float a_maxzoom;\\nattribute float a_rangeend;\\nattribute float a_rangestart;\\nattribute float a_labelminzoom;\\nuniform mat4 u_posmatrix;\\nuniform mat4 u_exmatrix;\\nuniform float u_angle;\\nuniform float u_zoom;\\nuniform float u_flip;\\nuniform float u_fadedist;\\nuniform float u_minfadezoom;\\nuniform float u_maxfadezoom;\\nuniform float u_fadezoom;\\nuniform vec2 u_texsize;\\nvarying vec2 a;\\nvarying float b;\\nvoid main ()\\n{\\n  float d_1;\\n  d_1 = 0.0;\\n  float tmpvar_2;\\n  tmpvar_2 = (float(mod ((a_angle + u_angle), 256.0)));\\n  if ((((u_flip > 0.0) && (tmpvar_2 >= 64.0)) && (tmpvar_2 < 192.0))) {\\n    d_1 = 1.0;\\n  };\\n  float tmpvar_3;\\n  tmpvar_3 = (((2.0 - \\n    float((u_zoom >= a_minzoom))\\n  ) - (1.0 - \\n    float((u_zoom >= a_maxzoom))\\n  )) + d_1);\\n  float tmpvar_4;\\n  tmpvar_4 = clamp (((u_fadezoom - a_labelminzoom) / u_fadedist), 0.0, 1.0);\\n  if ((u_fadedist >= 0.0)) {\\n    b = tmpvar_4;\\n  } else {\\n    b = (1.0 - tmpvar_4);\\n  };\\n  if ((u_maxfadezoom < a_labelminzoom)) {\\n    b = 0.0;\\n  };\\n  if ((u_minfadezoom >= a_labelminzoom)) {\\n    b = 1.0;\\n  };\\n  vec4 tmpvar_5;\\n  tmpvar_5.zw = vec2(0.0, 1.0);\\n  tmpvar_5.xy = a_pos;\\n  vec4 tmpvar_6;\\n  tmpvar_6.w = 0.0;\\n  tmpvar_6.xy = (a_offset / 64.0);\\n  tmpvar_6.z = ((tmpvar_3 + float(\\n    (0.0 >= b)\\n  )) + (float(\\n    (u_angle >= a_rangeend)\\n  ) * (1.0 - \\n    float((u_angle >= a_rangestart))\\n  )));\\n  gl_Position = ((u_posmatrix * tmpvar_5) + (u_exmatrix * tmpvar_6));\\n  a = (a_tex / u_texsize);\\n}\\n\\n\",\"fragment\":\"precision mediump float;\\nuniform sampler2D u_texture;\\nuniform vec4 u_color;\\nuniform float u_buffer;\\nuniform float u_gamma;\\nvarying vec2 a;\\nvarying float b;\\nvoid main ()\\n{\\n  float edge0_1;\\n  edge0_1 = (u_buffer - u_gamma);\\n  lowp float tmpvar_2;\\n  tmpvar_2 = clamp (((texture2D (u_texture, a).w - edge0_1) / (\\n    (u_buffer + u_gamma)\\n   - edge0_1)), 0.0, 1.0);\\n  lowp vec4 tmpvar_3;\\n  tmpvar_3 = (u_color * ((tmpvar_2 * \\n    (tmpvar_2 * (3.0 - (2.0 * tmpvar_2)))\\n  ) * b));\\n  gl_FragColor = tmpvar_3;\\n}\\n\\n\"}\n};\n","'use strict';\n\nvar Source = require('./source.js');\nvar GeoJSONTile = require('./geojsontile.js');\n\nvar GeoJSONSource = module.exports = function(options) {\n    this.tiles = {};\n    this.alltiles = {};\n    this.enabled = true;\n\n    this.zooms = [1, 5, 9, 13];\n    this.minTileZoom = this.zooms[0];\n    this.maxTileZoom = this.zooms[this.zooms.length - 1];\n\n    this.loadNewTiles = true;\n    this.tileJSON = {\n        minZoom: 1,\n        maxZoom: 13\n    };\n\n    this.data = options.data;\n};\n\nGeoJSONSource.prototype = Object.create(Source.prototype);\n\nGeoJSONSource.prototype.setData = function(data) {\n    this.data = data;\n    if (this.map) this._updateData();\n    return this;\n};\n\nGeoJSONSource.prototype.onAdd = function(map) {\n    this.map = map;\n    this.painter = map.painter;\n\n    if (this.map.style) this._updateData();\n    map.on('style.change', this._updateData.bind(this));\n};\n\nGeoJSONSource.prototype._updateData = function() {\n    var source = this;\n    this.workerID = this.map.dispatcher.send('parse geojson', {\n        data: this.data,\n        zooms: this.zooms,\n        tileSize: 512,\n        source: this.id\n    }, function(err, tiles) {\n        if (err) return;\n        for (var i = 0; i < tiles.length; i++) {\n            source.alltiles[tiles[i].id] = new GeoJSONTile(tiles[i].id, source, tiles[i]);\n        }\n        if (source.map) source.map.update();\n    });\n    return this;\n};\n\nGeoJSONSource.prototype._addTile = function(id) {\n    var tile = this.alltiles[id];\n    if (tile) {\n        tile._load();\n        this.tiles[id] = tile;\n        this.fire('tile.add', {tile: tile});\n    }\n    return tile || {};\n};\n\nGeoJSONSource.prototype._coveringZoomLevel = function(zoom) {\n    for (var i = this.zooms.length - 1; i >= 0; i--) {\n        if (this.zooms[i] <= zoom) {\n            var z = this.zooms[i];\n            return z;\n        }\n    }\n    return 0;\n};\n","'use strict';\n\nvar Tile = require('./tile.js');\nvar BufferSet = require('../data/buffer/bufferset.js');\nvar createBucket = require('../data/createbucket.js');\n\nmodule.exports = GeoJSONTile;\n\nfunction GeoJSONTile(id, source, data) {\n    this.id = id;\n    this.source = source;\n    this.data = data;\n    this.workerID = source.workerID;\n}\n\nGeoJSONTile.prototype = Object.create(Tile.prototype);\n\nGeoJSONTile.prototype._load = function() {\n    if (this.loaded) return;\n    this.loaded = true;\n\n    var data = this.data;\n    this.buffers = new BufferSet(data.buffers);\n\n    this.buckets = {};\n    for (var b in data.elementGroups) {\n        this.buckets[b] = createBucket(this.source.map.style.buckets[b], this.buffers, undefined, data.elementGroups[b]);\n    }\n\n\n};\n\n// noops\nGeoJSONTile.prototype.abort = function() { };\nGeoJSONTile.prototype.remove = function() { };\n","'use strict';\n\nmodule.exports = Wrapper;\n\n// conform to vectortile api\nfunction Wrapper(features) {\n    this.features = features;\n    this.length = features.length;\n}\n\nWrapper.prototype.feature = function(i) {\n    return new FeatureWrapper(this.features[i]);\n};\n\nvar mapping = {\n    'Point': 1,\n    'LineString': 2,\n    'Polygon': 3\n};\n\nfunction FeatureWrapper(feature) {\n    this.feature = feature;\n    this._type = mapping[feature.type];\n    this.properties = feature.properties;\n}\n\nFeatureWrapper.prototype.loadGeometry = function() {\n    return this.feature.coords;\n};\n\nFeatureWrapper.prototype.bbox = function() {\n\n    if (this._type === mapping.Point) {\n        return [\n            this.feature.coords[0],\n            this.feature.coords[1],\n            this.feature.coords[0],\n            this.feature.coords[1]\n        ];\n    }\n\n    var rings = this.feature.coords;\n\n    var x1 = Infinity,\n        x2 = -Infinity,\n        y1 = Infinity,\n        y2 = -Infinity;\n\n    for (var i = 0; i < rings.length; i++) {\n        var ring = rings[i];\n\n        for (var j = 0; j < ring.length; j++) {\n            var coord = ring[j];\n\n            x1 = Math.min(x1, coord.x);\n            x2 = Math.max(x2, coord.x);\n            y1 = Math.min(y1, coord.y);\n            y2 = Math.max(y2, coord.y);\n        }\n    }\n\n    return [x1, y1, x2, y2];\n};\n","'use strict';\n\nvar Tile = require('./tile.js');\nvar ajax = require('../util/ajax.js');\n\nmodule.exports = RasterTile;\n\nfunction RasterTile(id, source, url, callback) {\n    this.id = id;\n    this.loaded = false;\n    this.url = url;\n    this.source = source;\n    this.map = source.map;\n    this._load();\n    this.callback = callback;\n    this.uses = 1;\n\n    // Todo finish figuring out what raster buckets are\n    this.buckets = {};\n    this.info = { raster: true };\n    var buckets = this.map.style.buckets;\n    for (var b in buckets) {\n        var bucket = buckets[b];\n        var sourceid = bucket && bucket.source;\n        if (source.id === sourceid) {\n            this.buckets[b] = {\n                info: bucket.render,\n                type: 'raster',\n                tile: this\n            };\n        }\n    }\n}\n\nRasterTile.prototype = Object.create(Tile.prototype);\n\nRasterTile.prototype._load = function() {\n    var tile = this;\n    ajax.getImage(this.url, function(err, img) {\n        // @TODO handle errors.\n        if (err) return;\n        tile.img = img;\n        if (tile.map) tile.onTileLoad();\n    });\n};\n\nRasterTile.prototype.onTileLoad = function() {\n    // start texture upload\n    this.bind(this.map.painter.gl);\n\n    this.loaded = true;\n    this.callback();\n};\n\nRasterTile.prototype.abort = function() {\n    this.aborted = true;\n    if (this.img) this.img.src = '';\n    delete this.img;\n};\n\nRasterTile.prototype.bind = function(gl) {\n    if (!this.texture) {\n        // try to find reusable texture\n        this.texture = this.map.painter.getTexture(this.img.width);\n        if (this.texture) {\n            gl.bindTexture(gl.TEXTURE_2D, this.texture);\n            gl.texSubImage2D(gl.TEXTURE_2D, 0, 0, 0, gl.RGBA, gl.UNSIGNED_BYTE, this.img);\n        } else {\n            this.texture = gl.createTexture();\n            gl.bindTexture(gl.TEXTURE_2D, this.texture);\n            gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR_MIPMAP_NEAREST);\n            gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR);\n            gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n            gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n            gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, this.img);\n            this.texture.size = this.img.width;\n        }\n        gl.generateMipmap(gl.TEXTURE_2D);\n    } else {\n        gl.bindTexture(gl.TEXTURE_2D, this.texture);\n    }\n};\n\nRasterTile.prototype.remove = function() {\n    if (this.texture) this.map.painter.saveTexture(this.texture);\n    delete this.map;\n};\n\nRasterTile.prototype.featuresAt = function(pos, params, callback) {\n    // noop\n    callback(null, []);\n};\n","'use strict';\n\nvar util = require('../util/util.js'),\n    ajax = require('../util/ajax.js'),\n    tileJSON = require('../util/url.js').tileJSON,\n    Evented = require('../util/evented.js'),\n    Cache = require('../util/mrucache.js'),\n    TileCoord = require('./tilecoord'),\n    VectorTile = require('./vectortile'),\n    RasterTile = require('./rastertile.js'),\n    Point = require('point-geometry');\n\nmodule.exports = Source;\n\nSource.protocols = {\n    \"mapbox\": function(url, callback) {\n        ajax.getJSON(tileJSON(url.split('://')[1]), callback);\n    }\n};\n\nfunction Source(options) {\n    this.tiles = {};\n    this.enabled = false;\n    this.type = options.type;\n    if (this.type === 'vector' && options.tileSize && options.tileSize !== 512) {\n        throw new Error('vector tile sources must have a tileSize of 512');\n    }\n    this.Tile = this.type === 'vector' ? VectorTile : RasterTile;\n    this.options = util.inherit(this.options, options);\n    this.cache = new Cache(this.options.cacheSize, function(tile) {\n        tile.remove();\n    });\n\n    var protocol = options.url.split(':')[0];\n    Source.protocols[protocol](options.url, function(err, tileJSON) {\n        if (err) throw err;\n        this.tileJSON = tileJSON;\n        this.loadNewTiles = true;\n        this.enabled = true;\n        this.update();\n\n        if (this.map) this.map.fire('source.add', {source: this});\n    }.bind(this));\n\n    this._updateTiles = util.throttle(this._updateTiles, 50, this);\n}\n\nSource.prototype = util.inherit(Evented, {\n    options: {\n        tileSize: 512,\n        cacheSize: 20\n    },\n\n    onAdd: function(map) {\n        this.map = map;\n        this.painter = map.painter;\n    },\n\n    load: function() {\n        for (var t in this.tiles) {\n            this.tiles[t]._load();\n        }\n    },\n\n    loaded: function() {\n        for (var t in this.tiles) {\n            if (!this.tiles[t].loaded)\n                return false;\n        }\n        return true;\n    },\n\n    update: function() {\n        if (!this.enabled) return;\n        this._updateTiles();\n    },\n\n    render: function(layers) {\n        // Iteratively paint every tile.\n        if (!this.enabled) return;\n        var order = Object.keys(this.tiles);\n        order.sort(this._z_order);\n        for (var i = 0; i < order.length; i++) {\n            var id = order[i];\n            var tile = this.tiles[id];\n            if (tile.loaded && !this.coveredTiles[id]) {\n                this._renderTile(tile, id, layers);\n            }\n        }\n    },\n\n    featuresAt: function(point, params, callback) {\n        point = Point.convert(point);\n\n        if (params.layer) {\n            var style = this.map.style,\n                layer = style.getLayer(params.layer);\n            params.bucket = style.buckets[layer.ref || layer.id];\n        }\n\n        var order = Object.keys(this.tiles);\n        order.sort(this._z_order);\n        for (var i = 0; i < order.length; i++) {\n            var id = order[i];\n            var tile = this.tiles[id];\n            var pos = tile.positionAt(id, point);\n\n            if (pos && pos.x >= 0 && pos.x < 4096 && pos.y >= 0 && pos.y < 4096) {\n                // The click is within the viewport. There is only ever one tile in\n                // a layer that has this property.\n                return tile.featuresAt(pos, params, callback);\n            }\n        }\n\n        callback(null, []);\n    },\n\n    // get the zoom level adjusted for the difference in map and source tilesizes\n    _getZoom: function() {\n        var zOffset = Math.log(this.map.transform.tileSize/this.options.tileSize) / Math.LN2;\n        return this.map.transform.zoom + zOffset;\n    },\n\n    _coveringZoomLevel: function(zoom) {\n        for (var z = this.tileJSON.maxzoom; z >= this.tileJSON.minzoom; z--) {\n            if (z <= zoom) {\n                if (this.type === 'raster') {\n                    // allow underscaling by rounding to the nearest zoom level\n                    if (z < this.tileJSON.maxzoom) {\n                        z += Math.round(zoom - z);\n                    }\n                }\n                return z;\n            }\n        }\n        return 0;\n    },\n\n    _childZoomLevel: function(zoom) {\n        zoom = Math.max(this.tileJSON.minzoom, zoom + 1);\n        return zoom <= this.tileJSON.maxzoom ? zoom : null;\n    },\n\n    _getCoveringTiles: function(zoom) {\n        if (zoom === undefined) zoom = this._getZoom();\n        var z = this._coveringZoomLevel(zoom),\n            tiles = 1 << z,\n            tr = this.map.transform,\n            tileCenter = TileCoord.zoomTo(tr.locationCoordinate(tr.center), z);\n\n        var points = [\n            TileCoord.zoomTo(tr.pointCoordinate(tileCenter, {x: 0, y: 0}), z),\n            TileCoord.zoomTo(tr.pointCoordinate(tileCenter, {x: tr.width, y: 0}), z),\n            TileCoord.zoomTo(tr.pointCoordinate(tileCenter, {x: tr.width, y: tr.height}), z),\n            TileCoord.zoomTo(tr.pointCoordinate(tileCenter, {x: 0, y: tr.height}), z)\n        ], t = {};\n\n        // Divide the screen up in two triangles and scan each of them:\n        // +---/\n        // | / |\n        // /---+\n        this._scanTriangle(points[0], points[1], points[2], 0, tiles, scanLine);\n        this._scanTriangle(points[2], points[3], points[0], 0, tiles, scanLine);\n\n        return Object.keys(t).sort(fromCenter);\n\n        function fromCenter(a, b) {\n            var ad = Math.abs(a.x - tileCenter.column) +\n                    Math.abs(a.y - tileCenter.row),\n                bd = Math.abs(b.x - tileCenter.column) +\n                    Math.abs(b.y - tileCenter.row);\n\n            return ad - bd;\n        }\n\n        function scanLine(x0, x1, y) {\n            var x, wx;\n            if (y >= 0 && y <= tiles) {\n                for (x = x0; x < x1; x++) {\n                    wx = (x + tiles) % tiles;\n                    t[TileCoord.toID(z, wx, y, Math.floor(x/tiles))] = {x: wx, y: y};\n                }\n            }\n        }\n    },\n\n    // Given a tile of data, its id, and a style layers, render the tile to the canvas\n    _renderTile: function(tile, id, layers) {\n        var pos = TileCoord.fromID(id);\n        var z = pos.z, x = pos.x, y = pos.y, w = pos.w;\n        x += w * (1 << z);\n\n        tile.calculateMatrices(z, x, y, this.map.transform, this.painter);\n\n        this.painter.draw(tile, this.map.style, layers, {\n            z: z, x: x, y: y,\n            debug: this.map.debug,\n            antialiasing: this.map.antialiasing,\n            vertices: this.map.vertices,\n            rotating: this.map.rotating,\n            zooming: this.map.zooming\n        });\n    },\n\n    // Recursively find children of the given tile (up to maxCoveringZoom) that are already loaded;\n    // adds found tiles to retain object; returns true if children completely cover the tile\n\n    _findLoadedChildren: function(id, maxCoveringZoom, retain) {\n        var complete = true;\n        var z = TileCoord.fromID(id).z;\n        var ids = TileCoord.children(id);\n        for (var i = 0; i < ids.length; i++) {\n            if (this.tiles[ids[i]] && this.tiles[ids[i]].loaded) {\n                retain[ids[i]] = true;\n            } else {\n                complete = false;\n                if (z < maxCoveringZoom) {\n                    // Go further down the hierarchy to find more unloaded children.\n                    this._findLoadedChildren(ids[i], maxCoveringZoom, retain);\n                }\n            }\n        }\n        return complete;\n    },\n\n    // Find a loaded parent of the given tile (up to minCoveringZoom);\n    // adds the found tile to retain object and returns true if a parent was found\n\n    _findLoadedParent: function(id, minCoveringZoom, retain) {\n        for (var z = TileCoord.fromID(id).z; z >= minCoveringZoom; z--) {\n            id = TileCoord.parent(id);\n            if (this.tiles[id] && this.tiles[id].loaded) {\n                retain[id] = true;\n                return true;\n            }\n        }\n        return false;\n    },\n\n    // Removes tiles that are outside the viewport and adds new tiles that are inside the viewport.\n    _updateTiles: function() {\n        if (!this.map || !this.map.loadNewTiles || !this.loadNewTiles || !this.map.style.sources[this.id]) return;\n\n        var zoom = Math.floor(this._getZoom());\n        var required = this._getCoveringTiles().sort(this._centerOut.bind(this));\n        var i;\n        var id;\n        var complete;\n        var tile;\n\n        // Determine the overzooming/underzooming amounts.\n        var minCoveringZoom = Math.max(this.tileJSON.minzoom, zoom - 10);\n        var maxCoveringZoom = this.tileJSON.minzoom;\n        while (maxCoveringZoom < zoom + 1) {\n            var level = this._childZoomLevel(maxCoveringZoom);\n            if (level === null) break;\n            else maxCoveringZoom = level;\n        }\n\n        // Retain is a list of tiles that we shouldn't delete, even if they are not\n        // the most ideal tile for the current viewport. This may include tiles like\n        // parent or child tiles that are *already* loaded.\n        var retain = {};\n        // Covered is a list of retained tiles who's areas are full covered by other,\n        // better, retained tiles. They are not drawn separately.\n        this.coveredTiles = {};\n\n        var fullyComplete = true;\n\n        // Add existing child/parent tiles if the actual tile is not yet loaded\n        for (i = 0; i < required.length; i++) {\n            id = +required[i];\n            retain[id] = true;\n            tile = this._addTile(id);\n\n            if (!tile.loaded) {\n                // The tile we require is not yet loaded. Try to find a parent or\n                // child tile that we already have.\n\n                // First, try to find existing child tiles that completely cover the\n                // missing tile.\n                complete = this._findLoadedChildren(id, maxCoveringZoom, retain);\n\n                // Then, if there are no complete child tiles, try to find existing\n                // parent tiles that completely cover the missing tile.\n                if (!complete) {\n                    complete = this._findLoadedParent(id, minCoveringZoom, retain);\n                }\n\n                // The unloaded tile's area is not completely covered loaded tiles\n                if (!complete) {\n                    fullyComplete = false;\n                }\n            }\n        }\n\n        var now = new Date().getTime();\n        var fadeDuration = this.type === 'raster' ? this.map.style.rasterFadeDuration : 0;\n\n        for (id in retain) {\n            tile = this.tiles[id];\n            if (tile && tile.timeAdded > now - fadeDuration) {\n                // This tile is still fading in. Find tiles to cross-fade with it.\n\n                complete = this._findLoadedChildren(id, maxCoveringZoom, retain);\n\n                if (complete) {\n                    this.coveredTiles[id] = true;\n                } else {\n                    this._findLoadedParent(id, minCoveringZoom, retain);\n                }\n            }\n        }\n\n        for (id in this.coveredTiles) retain[id] = true;\n\n        // Remove the tiles we don't need anymore.\n        var remove = util.keysDifference(this.tiles, retain);\n        for (i = 0; i < remove.length; i++) {\n            id = +remove[i];\n            this._removeTile(id);\n        }\n    },\n\n    _loadTile: function(id) {\n        var layer = this;\n        var map = this.map,\n            pos = TileCoord.fromID(id),\n            tile;\n\n        if (pos.w === 0) {\n            // console.time('loading ' + pos.z + '/' + pos.x + '/' + pos.y);\n            var url = TileCoord.url(id, this.tileJSON.tiles);\n            tile = this.tiles[id] = new this.Tile(id, this, url, tileComplete);\n        } else {\n            var wrapped = TileCoord.toID(pos.z, pos.x, pos.y, 0);\n            tile = this.tiles[id] = this.tiles[wrapped] || this._addTile(wrapped);\n            tile.uses++;\n        }\n\n        function tileComplete(err) {\n            // console.timeEnd('loading ' + pos.z + '/' + pos.x + '/' + pos.y);\n            if (err) {\n                console.warn('failed to load tile %d/%d/%d: %s', pos.z, pos.x, pos.y, err.stack || err);\n            } else {\n                layer.fire('tile.load', {tile: tile});\n                map.update();\n            }\n        }\n\n        return tile;\n    },\n\n    // Adds a vector tile to the map. It will trigger a rerender of the map and will\n    // be part in all future renders of the map. The map object will handle copying\n    // the tile data to the GPU if it is required to paint the current viewport.\n    _addTile: function(id) {\n        var tile = this.tiles[id];\n\n        if (!tile) {\n            tile = this.cache.get(id);\n            if (tile) {\n                tile.uses = 1;\n                this.tiles[id] = tile;\n            }\n        }\n\n        if (!tile) {\n            tile = this._loadTile(id);\n            this.fire('tile.add', {tile: tile});\n        }\n\n        if (tile && tile.loaded && !tile.timeAdded) {\n            tile.timeAdded = new Date().getTime();\n            if (this.type === 'raster') {\n                this.map.animationLoop.set(this.map.style.rasterFadeDuration);\n            }\n        }\n\n        return tile;\n    },\n\n    _removeTile: function(id) {\n        var tile = this.tiles[id];\n        if (tile) {\n            tile.uses--;\n            delete this.tiles[id];\n\n            if (tile.uses <= 0) {\n                delete tile.timeAdded;\n                if (!tile.loaded) {\n                    tile.abort();\n                    tile.remove();\n                } else {\n                    this.cache.add(id, tile);\n                }\n\n                this.fire('tile.remove', {tile: tile});\n            }\n        }\n    },\n\n    // Taken from polymaps src/Layer.js\n    // https://github.com/simplegeo/polymaps/blob/master/src/Layer.js#L333-L383\n\n    // scan-line conversion\n    _scanTriangle: function(a, b, c, ymin, ymax, scanLine) {\n        var ab = this._edge(a, b),\n            bc = this._edge(b, c),\n            ca = this._edge(c, a);\n\n        var t;\n\n        // sort edges by y-length\n        if (ab.dy > bc.dy) { t = ab; ab = bc; bc = t; }\n        if (ab.dy > ca.dy) { t = ab; ab = ca; ca = t; }\n        if (bc.dy > ca.dy) { t = bc; bc = ca; ca = t; }\n\n        // scan span! scan span!\n        if (ab.dy) this._scanSpans(ca, ab, ymin, ymax, scanLine);\n        if (bc.dy) this._scanSpans(ca, bc, ymin, ymax, scanLine);\n    },\n\n    // scan-line conversion\n    _edge: function(a, b) {\n        if (a.row > b.row) { var t = a; a = b; b = t; }\n        return {\n            x0: a.column,\n            y0: a.row,\n            x1: b.column,\n            y1: b.row,\n            dx: b.column - a.column,\n            dy: b.row - a.row\n        };\n    },\n\n    // scan-line conversion\n    _scanSpans: function(e0, e1, ymin, ymax, scanLine) {\n        var y0 = Math.max(ymin, Math.floor(e1.y0)),\n            y1 = Math.min(ymax, Math.ceil(e1.y1));\n\n        // sort edges by x-coordinate\n        if ((e0.x0 == e1.x0 && e0.y0 == e1.y0) ?\n            (e0.x0 + e1.dy / e0.dy * e0.dx < e1.x1) :\n            (e0.x1 - e1.dy / e0.dy * e0.dx < e1.x0)) {\n            var t = e0; e0 = e1; e1 = t;\n        }\n\n        // scan lines!\n        var m0 = e0.dx / e0.dy,\n            m1 = e1.dx / e1.dy,\n            d0 = e0.dx > 0, // use y + 1 to compute x0\n            d1 = e1.dx < 0; // use y + 1 to compute x1\n        for (var y = y0; y < y1; y++) {\n            var x0 = m0 * Math.max(0, Math.min(e0.dy, y + d0 - e0.y0)) + e0.x0,\n                x1 = m1 * Math.max(0, Math.min(e1.dy, y + d1 - e1.y0)) + e1.x0;\n            scanLine(Math.floor(x1), Math.ceil(x0), y);\n        }\n    },\n\n    _z_order: function(a, b) {\n        return (b % 32) - (a % 32);\n    },\n\n    _centerOut: function(a, b) {\n        var tr = this.map.transform;\n        var aPos = TileCoord.fromID(a);\n        var bPos = TileCoord.fromID(b);\n        var c = TileCoord.zoomTo(tr.locationCoordinate(tr.center), aPos.z);\n        var center = new Point(c.column - 0.5, c.row - 0.5);\n        return center.dist(aPos) - center.dist(bPos);\n    },\n});\n\nvar sources = {\n    vector: Source,\n    raster: Source,\n    geojson: require('./geojsonsource'),\n    video: require('./videosource')\n};\n\nSource.create = function(source) {\n    return new sources[source.type](source);\n};\n","'use strict';\n\nvar glmatrix = require('../lib/glmatrix.js'),\n    mat2 = glmatrix.mat2,\n    mat4 = glmatrix.mat4,\n    vec2 = glmatrix.vec2;\n\nmodule.exports = Tile;\n\nfunction Tile() {}\n\nTile.prototype = {\n    // todo unhardcode\n    tileExtent: 4096,\n\n    calculateMatrices: function(z, x, y, transform, painter) {\n\n        // Initialize model-view matrix that converts from the tile coordinates\n        // to screen coordinates.\n        var tileScale = Math.pow(2, z);\n        var scale = transform.worldSize / tileScale;\n\n        // TODO: remove\n        this.scale = scale;\n\n        // The position matrix\n        this.posMatrix = mat4.create();\n        mat4.translate(this.posMatrix, this.posMatrix, [transform.centerPoint.x, transform.centerPoint.y, 0]);\n        mat4.rotateZ(this.posMatrix, this.posMatrix, transform.angle);\n        mat4.translate(this.posMatrix, this.posMatrix, [-transform.centerPoint.x, -transform.centerPoint.y, 0]);\n\n        var pixelX = transform.width / 2 - transform.x,\n            pixelY = transform.height / 2 - transform.y;\n\n        mat4.translate(this.posMatrix, this.posMatrix, [pixelX + x * scale, pixelY + y * scale, 1]);\n\n        // Create inverted matrix for interaction\n        this.invPosMatrix = mat4.create();\n        mat4.invert(this.invPosMatrix, this.posMatrix);\n\n        mat4.scale(this.posMatrix, this.posMatrix, [ scale / this.tileExtent, scale / this.tileExtent, 1 ]);\n        mat4.multiply(this.posMatrix, painter.projectionMatrix, this.posMatrix);\n\n        // The extrusion matrix.\n        this.exMatrix = mat4.clone(painter.projectionMatrix);\n        mat4.rotateZ(this.exMatrix, this.exMatrix, transform.angle);\n\n        // 2x2 matrix for rotating points\n        this.rotationMatrix = mat2.create();\n        mat2.rotate(this.rotationMatrix, this.rotationMatrix, transform.angle);\n    },\n\n    positionAt: function(id, point) {\n        // tile hasn't finished loading\n        if (!this.invPosMatrix) return null;\n\n        var pos = vec2.transformMat4([], [point.x, point.y], this.invPosMatrix);\n        vec2.scale(pos, pos, 4096 / this.scale);\n        return {\n            x: pos[0],\n            y: pos[1],\n            scale: this.scale\n        };\n    },\n\n    featuresAt: function(pos, params, callback) {\n        this.source.map.dispatcher.send('query features', {\n            id: this.id,\n            x: pos.x,\n            y: pos.y,\n            scale: pos.scale,\n            source: this.source.id,\n            params: params\n        }, callback, this.workerID);\n    }\n};\n","'use strict';\n\n/*\n * Tiles are generally represented as packed integer ids constructed by\n * `TileCoord.toID(x, y, z)`\n */\n\nvar TileCoord = exports;\n\nTileCoord.toID = function(z, x, y, w) {\n    w = w || 0;\n    w *= 2;\n    if (w < 0) w = w * -1 -1;\n    var dim = 1 << z;\n    return ((dim * dim * w + dim * y + x) * 32) + z;\n};\n\nTileCoord.asString = function(id) {\n    var pos = TileCoord.fromID(id);\n    return pos.z + \"/\" + pos.x + \"/\" + pos.y;\n};\n\n/*\n * Parse a packed integer id into an object with x, y, and z properties\n */\nTileCoord.fromID = function(id) {\n    var z = id % 32, dim = 1 << z;\n    var xy = ((id - z) / 32);\n    var x = xy % dim, y = ((xy - x) / dim) % dim;\n    var w = Math.floor(xy / (dim * dim));\n    if (w % 2 !== 0) w = w * -1 -1;\n    w /= 2;\n    return { z: z, x: x, y: y, w: w };\n};\n\n/*\n * Given a packed integer id, return its zoom level\n */\nTileCoord.zoom = function(id) {\n    return id % 32;\n};\n\n// Given an id and a list of urls, choose a url template and return a tile URL\nTileCoord.url = function(id, urls) {\n    var pos = TileCoord.fromID(id);\n\n    return urls[Math.floor((pos.x + pos.y) % urls.length)]\n        .replace('{h}', (pos.x % 16).toString(16) + (pos.y % 16).toString(16))\n        .replace('{z}', pos.z.toFixed(0))\n        .replace('{x}', pos.x.toFixed(0))\n        .replace('{y}', pos.y.toFixed(0));\n};\n\n/*\n * Given a packed integer id, return the id of its parent tile\n */\nTileCoord.parent = function(id) {\n    var pos = TileCoord.fromID(id);\n    if (pos.z === 0) return id;\n    else return TileCoord.toID(pos.z - 1, Math.floor(pos.x / 2), Math.floor(pos.y / 2));\n};\n\nTileCoord.parentWithZoom = function(id, zoom) {\n    var pos = TileCoord.fromID(id);\n    while (pos.z > zoom) {\n        pos.z--;\n        pos.x = Math.floor(pos.x / 2);\n        pos.y = Math.floor(pos.y / 2);\n    }\n    return TileCoord.toID(pos.z, pos.x, pos.y);\n};\n\n/*\n * Given a packed integer id, return an array of integer ids representing\n * its four children.\n */\nTileCoord.children = function(id) {\n    var pos = TileCoord.fromID(id);\n    pos.z += 1;\n    pos.x *= 2;\n    pos.y *= 2;\n    return [\n        TileCoord.toID(pos.z, pos.x, pos.y, pos.w),\n        TileCoord.toID(pos.z, pos.x + 1, pos.y, pos.w),\n        TileCoord.toID(pos.z, pos.x, pos.y + 1, pos.w),\n        TileCoord.toID(pos.z, pos.x + 1, pos.y + 1, pos.w)\n    ];\n};\n\nTileCoord.zoomTo = function(c, z) {\n    c.column = c.column * Math.pow(2, z - c.zoom);\n    c.row = c.row * Math.pow(2, z - c.zoom);\n    c.zoom = z;\n    return c;\n};\n\n","'use strict';\n\nvar rewind = require('geojson-rewind');\n\nvar TileCoord = require('./tilecoord.js');\nvar Transform = require('../geo/transform.js');\nvar Point = require('point-geometry');\nvar LatLng = require('../geo/latlng.js');\n\nmodule.exports = tileGeoJSON;\n\nfunction tileGeoJSON(geojson, zoom) {\n    var tiles = {};\n    var tileExtent = 4096;\n    var transform = new Transform();\n    transform.zoom = zoom;\n\n    geojson = rewind(geojson);\n\n    if (geojson.type === 'FeatureCollection') {\n        for (var i = 0; i < geojson.features.length; i++) {\n            tileFeature(geojson.features[i], transform, tiles, tileExtent);\n        }\n\n    } else if (geojson.type === 'Feature') {\n        tileFeature(geojson, transform, tiles, tileExtent);\n\n    } else {\n        throw('Unrecognized geojson type');\n    }\n\n    return tiles;\n}\n\nfunction tileFeature(feature, transform, tiles, tileExtent) {\n    var coords = feature.geometry.coordinates;\n    var type = feature.geometry.type;\n\n    var tiled;\n    if (type === 'Point') {\n        tiled = tileLineString([coords], transform, tileExtent);\n\n    } else if (type === 'LineString' || type === 'MultiPoint') {\n        tiled = tileLineString(coords, transform, tileExtent);\n\n    } else if (type === 'Polygon' || type === 'MultiLineString') {\n        tiled = {};\n        for (var i = 0; i < coords.length; i++) {\n            var tiled_ = tileLineString(coords[i], transform, tileExtent, type === 'Polygon');\n            for (var tileID in tiled_) {\n                if (!tiled[tileID]) tiled[tileID] = [];\n                tiled[tileID] = (tiled[tileID] || []).concat(tiled_[tileID]);\n            }\n        }\n\n    } else if (type === 'MultiPolygon') {\n        throw(\"todo\");\n    } else {\n        throw(\"unrecognized geometry type\");\n    }\n\n    for (var id in tiled) {\n        tiles[id] = tiles[id] || [];\n        tiles[id].push({\n            properties: feature.properties,\n            coords: tiled[id],\n            type: feature.geometry.type\n        });\n    }\n}\n\nfunction tileLineString(coords, transform, tileExtent, rejoin) {\n\n    var padding = 0.01;\n    var paddedExtent = tileExtent * (1 + 2 * padding);\n    var coord = transform.locationCoordinate(new LatLng(coords[0][1], coords[0][0]));\n    var prevCoord;\n\n    var tiles = {};\n\n    for (var i = 0; i < coords.length; i++) {\n        prevCoord = coord;\n        coord = transform.locationCoordinate(new LatLng(coords[i][1], coords[i][0]));\n\n        var dx = coord.column - prevCoord.column || Number.MIN_VALUE,\n            dy = coord.row - prevCoord.row || Number.MIN_VALUE,\n            dirX = dx / Math.abs(dx),\n            dirY = dy / Math.abs(dy);\n\n        // Find the rectangular bounding box, in tiles, of the polygon\n        var startTileX = Math.floor(prevCoord.column - dirX * padding);\n        var endTileX = Math.floor(coord.column + dirX * padding);\n        var startTileY = Math.floor(prevCoord.row - dirY * padding);\n        var endTileY = Math.floor(coord.row + dirY * padding);\n\n        // Iterate over all tiles the segment might intersect\n        // and split the segment across those tiles\n        for (var x = startTileX; (x - endTileX) * dirX <= 0; x += dirX) {\n            var leftX = (x - padding - prevCoord.column) / dx;\n            var rightX = (x + 1 + padding - prevCoord.column) / dx;\n\n            for (var y = startTileY; (y - endTileY) * dirY <= 0; y += dirY) {\n                var topY = (y - padding - prevCoord.row) / dy;\n                var bottomY = (y + 1 + padding - prevCoord.row) / dy;\n\n                // fraction of the distance along the segment at which the segment\n                // enters or exits the tile\n                var enter = Math.max(Math.min(leftX, rightX), Math.min(topY, bottomY));\n                var exit = Math.min(Math.max(leftX, rightX), Math.max(topY, bottomY));\n\n                var tileID = TileCoord.toID(transform.tileZoom, x, y),\n                    tile = tiles[tileID],\n                    point;\n\n                // segments starts outside the tile, add entry point\n                if (0 <= enter && enter < 1) {\n                    point = new Point(\n                        ((prevCoord.column + enter * dx) - x) * tileExtent,\n                        ((prevCoord.row + enter * dy) - y) * tileExtent);\n\n                    point.continues = true;\n\n                    if (!tile) tiles[tileID] = tile = [];\n                    tile.push([point]);\n                }\n\n                // segments ends outside the tile, add exit point\n                if (0 <= exit && exit < 1) {\n                    point = new Point(\n                        ((prevCoord.column + exit * dx) - x) * tileExtent,\n                        ((prevCoord.row + exit * dy) - y) * tileExtent);\n\n                    point.continues = true;\n\n                    tile[tile.length - 1].push(point);\n\n                // add the point itself\n                } else {\n                    point = new Point(\n                        (coord.column - x) * tileExtent,\n                        (coord.row - y) * tileExtent);\n\n                    if (!tile) tiles[tileID] = tile = [[point]];\n                    else tile[tile.length - 1].push(point);\n                }\n            }\n        }\n    }\n\n    if (rejoin) {\n        // reassemble the disconnected segments into a linestring\n        // sections of the linestring outside the tile are replaced with segments\n        // that follow the tile's edge\n        for (var id in tiles) {\n\n            var segments = tiles[id];\n\n            if (!segments[0][0].continues && segments.length > 1) {\n                // if the first segment is the beginning of the linestring\n                // then join it with the last so that all segments start and\n                // end at tile boundaries\n                var last = segments.pop();\n                Array.prototype.unshift.apply(segments[0], last.slice(0, last.length - 1));\n            }\n\n            var start = edgeDist(segments[0][0], tileExtent, padding);\n\n            for (var k = 0; k < segments.length; k++) {\n                // Add all tile corners along the path between the current segment's exit point\n                // and the next segment's entry point\n\n                var thisExit = edgeDist(segments[k][segments[k].length - 1], paddedExtent);\n                var nextEntry = edgeDist(segments[(k + 1) % segments.length][0], paddedExtent);\n\n                var startToExit = (thisExit - start + 4) % 4;\n                var startToNextEntry = (nextEntry - start + 4) % 4;\n                var direction = (thisExit === nextEntry || startToExit < startToNextEntry) ? 1 : -1;\n                var roundFn = direction > 0 ? Math.ceil : Math.floor;\n\n                for (var c = roundFn(thisExit) % 4; c != roundFn(nextEntry) % 4; c = (c + direction + 4) % 4) {\n                    var corner = corners[c];\n                    segments[k].push(new Point(\n                        (corner.x + (corner.x - 0.5 > 0 ? 1 : -1) * padding) * tileExtent,\n                        (corner.y + (corner.y - 0.5 > 0 ? 1 : -1) * padding) * tileExtent));\n                }\n            }\n\n            // Join all segments\n            tiles[id] = [Array.prototype.concat.apply([], segments)];\n        }\n    }\n\n    return tiles;\n\n}\n\nvar corners = [\n    new Point(0, 0),\n    new Point(1, 0),\n    new Point(1, 1),\n    new Point(0, 1)];\n\n/*\n * Converts to a point to the distance along the edge of the tile (out of 4).\n *\n *         0.5\n *     0 _______ 1\n *      |       |\n *  3.5 |       | 1.5\n *      |       |\n *      |_______|\n *     3   2.5   2\n */\nfunction edgeDist(point, extent) {\n    var x = point.x / extent;\n    var y = point.y / extent;\n    var d;\n    if (Math.abs(y - 0.5) >= Math.abs(x - 0.5)) {\n        d = Math.round(y) * 2 + (y < 0.5 ? x : 1 - x);\n    } else {\n        d = Math.round(1 - x) * 2 + (x > 0.5 ? y : 1 - y) + 1;\n    }\n\n    return d % 4;\n}\n","'use strict';\n\nvar Tile = require('./tile.js'),\n    TileCoord = require('./tilecoord.js'),\n    BufferSet = require('../data/buffer/bufferset.js'),\n    util = require('../util/util.js');\n\nvar createBucket = require('../data/createbucket.js');\n\nmodule.exports = VectorTile;\n\nfunction VectorTile(id, source, url, callback) {\n    this.id = id;\n    this.loaded = false;\n    this.url = url;\n    this.zoom = TileCoord.fromID(id).z;\n    this.map = source.map;\n    this.options = source.options;\n    this.id = util.uniqueId();\n    this.callback = callback;\n    this.source = source;\n\n    if (this.zoom >= source.tileJSON.maxzoom) {\n        this.depth = this.map.options.maxZoom - this.zoom;\n    } else {\n        this.depth = 1;\n    }\n    this.uses = 1;\n    this._load();\n}\n\nVectorTile.prototype = util.inherit(Tile, {\n\n    _load: function() {\n        var tile = this;\n        this.workerID = this.map.dispatcher.send('load tile', {\n            url: this.url,\n            id: this.id,\n            zoom: this.zoom,\n            maxZoom: this.source.tileJSON.maxzoom,\n            tileSize: this.options.tileSize,\n            source: this.source.id,\n            depth: this.depth\n        }, function(err, data) {\n            if (!err && data) {\n                tile.onTileLoad(data);\n            }\n            tile.callback(err);\n        });\n    },\n\n    onTileLoad: function(data) {\n\n        // Tile has been removed from the map\n        if (!this.map) return;\n\n        this.buffers = new BufferSet(data.buffers);\n\n        this.buckets = {};\n        for (var b in data.elementGroups) {\n            this.buckets[b] = createBucket(this.map.style.buckets[b], this.buffers, undefined, data.elementGroups[b]);\n        }\n\n        this.loaded = true;\n    },\n\n    remove: function() {\n\n        // reuse prerendered textures\n        for (var bucket in this.buckets) {\n            if (this.buckets[bucket].prerendered) this.map.painter.saveTexture(this.buckets[bucket].prerendered.texture);\n        }\n\n        this.map.dispatcher.send('remove tile', { id: this.id, source: this.source.id }, null, this.workerID);\n        this.map.painter.glyphAtlas.removeGlyphs(this.id);\n\n        var gl = this.map.painter.gl;\n        var buffers = this.buffers;\n        if (buffers) {\n            for (var b in buffers) {\n                buffers[b].destroy(gl);\n            }\n        }\n        delete this.map;\n    },\n\n    abort: function() {\n        this.map.dispatcher.send('abort tile', { id: this.id, source: this.source.id }, null, this.workerID);\n    }\n});\n","'use strict';\n\nvar Tile = require('./tile.js');\nvar TileCoord = require('./tilecoord.js');\nvar LatLng = require('../geo/latlng.js');\nvar Point = require('point-geometry');\n\nmodule.exports = VideoSource;\n\nfunction VideoSource(options) {\n\n    this.video = document.createElement('video');\n    this.video.crossOrigin = 'Anonymous';\n    this.video.loop = true;\n\n    var urls = (typeof options.url === 'string') ? [options.url] : options.url;\n\n    for (var i = 0; i < urls.length; i++) {\n        var s = document.createElement('source');\n        s.src = urls[i];\n        this.video.appendChild(s);\n    }\n\n    this.coordinates = options.coordinates;\n    this.enabled = true;\n\n    var loopID;\n    var source = this;\n\n    // start repainting when video starts playing\n    this.video.addEventListener('playing', function() {\n        loopID = source.map.style.animationLoop.set(Infinity);\n        source.map._rerender();\n    });\n\n    // stop repainting when video stops\n    this.video.addEventListener('pause', function() {\n        source.map.style.animationLoop.cancel(loopID);\n    });\n\n}\n\nVideoSource.prototype.onAdd = function(map) {\n    this.map = map;\n    this.video.play();\n    this.createTile();\n};\n\nVideoSource.prototype.createTile = function() {\n    /*\n     * Calculate which mercator tile is suitable for rendering the video in\n     * and create a buffer with the corner coordinates. These coordinates\n     * may be outside the tile, because raster tiles aren't clipped when rendering.\n     */\n    var map = this.map;\n    var coords = this.coordinates.map(function(latlng) {\n        var loc = LatLng.convert(latlng);\n        return TileCoord.zoomTo(map.transform.locationCoordinate(loc), 0);\n    });\n\n    var minX = Infinity;\n    var minY = Infinity;\n    var maxX = -Infinity;\n    var maxY = -Infinity;\n\n    for (var i = 0; i < coords.length; i++) {\n        minX = Math.min(minX, coords[i].column);\n        minY = Math.min(minY, coords[i].row);\n        maxX = Math.max(maxX, coords[i].column);\n        maxY = Math.max(maxY, coords[i].row);\n    }\n\n    var dx = maxX - minX;\n    var dy = maxY - minY;\n    var dMax = Math.max(dx, dy);\n    var center = TileCoord.zoomTo({\n        column: (minX + maxX) / 2,\n        row: (minY + maxY) / 2,\n        zoom: 0\n    }, Math.floor(-Math.log(dMax) / Math.LN2));\n\n    var tileExtent = 4096;\n    var tileCoords = coords.map(function(coord) {\n        var zoomedCoord = TileCoord.zoomTo(coord, center.zoom);\n        return new Point(\n            Math.round((zoomedCoord.column - center.column) * tileExtent),\n            Math.round((zoomedCoord.row - center.row) * tileExtent));\n    });\n\n    var gl = map.painter.gl;\n    var maxInt16 = 32767;\n    var array = new Int16Array([\n        tileCoords[0].x, tileCoords[0].y, 0, 0,\n        tileCoords[1].x, tileCoords[1].y, maxInt16, 0,\n        tileCoords[3].x, tileCoords[3].y, 0, maxInt16,\n        tileCoords[2].x, tileCoords[2].y, maxInt16, maxInt16\n    ]);\n    this.boundsBuffer = gl.createBuffer();\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.boundsBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, array, gl.STATIC_DRAW);\n\n    this.tile = new Tile();\n    this.center = center;\n};\n\nVideoSource.prototype.load = function() {\n    // noop\n};\n\nVideoSource.prototype.update = function() {\n    // noop\n};\n\nVideoSource.prototype.render = function(layers) {\n    if (!this.enabled) return;\n    if (this.video.readyState < 2) return; // not enough data for current position\n\n    var layer = layers[0];\n\n    var bucket = {\n        type: 'raster',\n        tile: this,\n        boundsBuffer: this.boundsBuffer,\n        bind: this.bind.bind(this)\n    };\n\n    var buckets = {};\n    buckets[layer.bucket] = bucket;\n\n    var c = this.center;\n    this.tile.calculateMatrices(c.zoom, c.column, c.row, this.map.transform, this.map.painter);\n    this.map.painter.tile = this.tile;\n    this.map.painter.applyStyle(layer, this.map.style, buckets, {});\n};\n\nVideoSource.prototype.bind = function(gl) {\n\n    if (!this.texture) {\n        this.texture = gl.createTexture();\n        gl.bindTexture(gl.TEXTURE_2D, this.texture);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR);\n        gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, this.video);\n\n    } else {\n        gl.bindTexture(gl.TEXTURE_2D, this.texture);\n        gl.texSubImage2D(gl.TEXTURE_2D, 0, 0, 0, gl.RGBA, gl.UNSIGNED_BYTE, this.video);\n    }\n\n};\n\n\nVideoSource.prototype.featuresAt = function(point, params, callback) {\n    // TODO return pixel?\n    return callback(null, []);\n};\n","'use strict';\n\nvar Actor = require('../util/actor.js'),\n    bucketFilter = require('../style/bucketfilter.js'),\n    WorkerTile = require('./workertile.js'),\n    tileGeoJSON = require('./tilegeojson.js'),\n    Wrapper = require('./geojsonwrapper.js'),\n    util = require('../util/util.js'),\n    queue = require('queue-async'),\n    ajax = require('../util/ajax.js');\n\nmodule.exports = Worker;\n\nfunction Worker(self) {\n    this.self = self;\n    this.actor = new Actor(self, this);\n}\n\nutil.extend(Worker.prototype, {\n    alert: function() {\n        this.self.postMessage({\n            type: 'alert message',\n            data: [].slice.call(arguments)\n        });\n    },\n\n    // Updates the style to use for this map.\n    'set buckets': function(data) {\n        var buckets = WorkerTile.buckets = data;\n        for (var i = 0; i < buckets.length; i++) {\n            var bucket = buckets[i];\n            bucket.compare = bucketFilter(bucket.filter);\n        }\n    },\n\n    'set glyphs': function(data) {\n        WorkerTile.prototype.glyphs = data;\n    },\n\n    /*\n     * Load and parse a tile at `url`, and call `callback` with\n     * (err, response)\n     *\n     * @param {string} url\n     * @param {function} callback\n     */\n    'load tile': function(params, callback) {\n        new WorkerTile(params.url, undefined, params.id, params.zoom, params.maxZoom, params.tileSize, params.source, params.depth, this.actor, callback);\n    },\n\n    /*\n     * Abort the request keyed under `url`\n     *\n     * @param {string} url\n     */\n    'abort tile': function(params) {\n        WorkerTile.cancel(params.id, params.source);\n    },\n\n    'remove tile': function(params) {\n        var id = params.id;\n        var source = params.source;\n        if (WorkerTile.loaded[source] && WorkerTile.loaded[source][id]) {\n            delete WorkerTile.loaded[source][id];\n        }\n    },\n\n    'parse geojson': function(params, callback) {\n        var data = params.data,\n            zooms = params.zooms,\n            len = zooms.length,\n            maxZoom = zooms[len - 1],\n            actor = this.actor,\n            q = queue();\n\n        function worker(id, tile, zoom, callback) {\n            new WorkerTile(undefined, new Wrapper(tile), id, zoom, maxZoom, params.tileSize, params.source, 4, actor, function(err, data) {\n                if (err) return callback(err);\n                data.id = id;\n                callback(null, data);\n            });\n        }\n\n        function tileData(err, data) {\n            if (err) throw err;\n            for (var i = 0; i < len; i++) {\n                var zoom = zooms[i];\n                var tiles = tileGeoJSON(data, zoom);\n                for (var id in tiles) {\n                    q.defer(worker, id, tiles[id], zoom);\n                }\n            }\n            q.awaitAll(callback);\n        }\n\n        if (typeof data === 'string') ajax.getJSON(data, tileData);\n        else tileData(null, data);\n    },\n\n    'query features': function(params, callback) {\n        var tile = WorkerTile.loaded[params.source] && WorkerTile.loaded[params.source][params.id];\n        if (tile) {\n            tile.featureTree.query(params, callback);\n        } else {\n            callback(null, []);\n        }\n    }\n});\n","'use strict';\n\nvar FeatureTree = require('../data/featuretree.js');\nvar Protobuf = require('pbf');\nvar vt = require('vector-tile');\nvar Collision = require('../symbol/collision.js');\nvar getArrayBuffer = require('../util/ajax.js').getArrayBuffer;\n\nvar BufferSet = require('../data/buffer/bufferset.js');\nvar createBucket = require('../data/createbucket.js');\n\nmodule.exports = WorkerTile;\nfunction WorkerTile(url, data, id, zoom, maxZoom, tileSize, source, depth, actor, callback) {\n    var tile = this;\n    this.id = id;\n    this.zoom = zoom;\n    this.maxZoom = maxZoom;\n    this.tileSize = tileSize;\n    this.source = source;\n    this.depth = depth;\n    this.buffers = new BufferSet();\n\n    function loaded(data) {\n        WorkerTile.loaded[source] = WorkerTile.loaded[source] || {};\n        WorkerTile.loaded[source][id] = tile;\n        tile.data = data;\n        tile.parse(data, actor, callback);\n    }\n\n    if (url) {\n        if (WorkerTile.loading[source] === undefined) WorkerTile.loading[source] = {};\n        WorkerTile.loading[source][id] = getArrayBuffer(url, function(err, data) {\n            delete WorkerTile.loading[source][id];\n            if (err) {\n                callback(err);\n            } else {\n                loaded(new vt.VectorTile(new Protobuf(new Uint8Array(data))));\n            }\n        });\n    } else {\n        loaded(data);\n    }\n}\n\nWorkerTile.cancel = function(id, sourceID) {\n    var source = WorkerTile.loading[sourceID];\n    if (source && source[id]) {\n        source[id].abort();\n        delete source[id];\n    }\n};\n\n// Stores tiles that are currently loading.\nWorkerTile.loading = {};\n\n// Stores tiles that are currently loaded.\nWorkerTile.loaded = {};\n\n// Stores the style information.\nWorkerTile.buckets = [];\n\n/*\n * Given tile data, parse raw vertices and data, create a vector\n * tile and parse it into ready-to-render vertices.\n *\n * @param {object} data\n * @param {function} respond\n */\nWorkerTile.prototype.parse = function(data, actor, callback) {\n    var tile = this;\n    var bucketInfo = WorkerTile.buckets;\n    this.callback = callback;\n\n    var tileExtent = 4096;\n    this.collision = new Collision(this.zoom, tileExtent, this.tileSize, this.depth);\n    this.featureTree = new FeatureTree(getGeometry, getType);\n\n    var buckets = this.buckets = sortTileIntoBuckets(this, data, bucketInfo);\n\n    var key, bucket;\n    var prevPlacementBucket;\n\n    var remaining = WorkerTile.buckets.length;\n\n    /*\n     *  The async parsing here is a bit tricky.\n     *  Some buckets depend on resources that may need to be loaded async (glyphs).\n     *  Some buckets need to be parsed in order (to get placement priorities right).\n     *\n     *  Dependencies calls are initiated first to get those rolling.\n     *  Buckets that don't need to be parsed in order, aren't to save time.\n     */\n\n    var orderedBuckets = WorkerTile.buckets;\n    for (var i = 0; i < orderedBuckets.length; i++) {\n        bucket = buckets[orderedBuckets[i].id];\n        if (!bucket) {\n            remaining--;\n            continue; // raster bucket, etc\n        }\n\n        var filter = bucket.info.filter;\n        if (filter && filter.source !== this.source) continue;\n\n        // Link buckets that need to be parsed in order\n        if (bucket.collision) {\n            if (prevPlacementBucket) {\n                prevPlacementBucket.next = bucket;\n            } else {\n                bucket.previousPlaced = true;\n            }\n            prevPlacementBucket = bucket;\n        }\n\n        if (bucket.getDependencies) {\n            bucket.getDependencies(this, actor, dependenciesDone(bucket));\n        }\n\n    }\n\n    // parse buckets where order doesn't matter and no dependencies\n    for (key in buckets) {\n        bucket = buckets[key];\n        if (!bucket.getDependencies && !bucket.collision) {\n            parseBucket(tile, bucket);\n        }\n    }\n\n    function dependenciesDone(bucket) {\n        return function(err) {\n            bucket.dependenciesLoaded = true;\n            parseBucket(tile, bucket, err);\n        };\n    }\n\n    function parseBucket(tile, bucket, skip) {\n        if (bucket.getDependencies && !bucket.dependenciesLoaded) return;\n        if (bucket.collision && !bucket.previousPlaced) return;\n\n        if (!skip) {\n            var now = Date.now();\n            if (bucket.type !== 'raster') bucket.addFeatures();\n            var time = Date.now() - now;\n            if (bucket.interactive) {\n                for (var i = 0; i < bucket.features.length; i++) {\n                    var feature = bucket.features[i];\n                    tile.featureTree.insert(feature.bbox(), bucket.name, feature);\n                }\n            }\n            if (typeof self !== 'undefined') {\n                self.bucketStats = self.bucketStats || {_total: 0};\n                self.bucketStats._total += time;\n                self.bucketStats[bucket.name] = (self.bucketStats[bucket.name] || 0) + time;\n            }\n        }\n\n        remaining--;\n        if (!remaining) return tile.done();\n\n        // try parsing the next bucket, if it is ready\n        if (bucket.next) {\n            bucket.next.previousPlaced = true;\n            parseBucket(tile, bucket.next);\n        }\n    }\n};\n\nWorkerTile.prototype.done = function() {\n    // Collect all buffers to mark them as transferable object.\n    var buffers = [];\n\n    for (var type in this.buffers) {\n        buffers.push(this.buffers[type].array);\n    }\n\n    // Convert buckets to a transferable format\n    var buckets = this.buckets;\n    var elementGroups = {};\n    for (var b in buckets) elementGroups[b] = buckets[b].elementGroups;\n\n    this.callback(null, {\n        elementGroups: elementGroups,\n        buffers: this.buffers\n    }, buffers);\n\n    // we don't need anything except featureTree at this point, so we mark it for GC\n    this.buffers = null;\n    this.collision = null;\n    this.buckets = null;\n};\n\nfunction sortTileIntoBuckets(tile, data, bucketInfo) {\n\n    var sourceLayers = {},\n        buckets = {},\n        layerName;\n\n    // For each source layer, find a list of buckets that use data from it\n    for (var i = 0; i < bucketInfo.length; i++) {\n        var info = bucketInfo[i];\n        var bucketName = info.id;\n\n        var minZoom = info['min-zoom'];\n        var maxZoom = info['max-zoom'];\n\n        if (info.source !== tile.source) continue;\n        if (minZoom && tile.zoom < minZoom && minZoom < tile.maxZoom) continue;\n        if (maxZoom && tile.zoom >= maxZoom) continue;\n\n        var bucket = createBucket(info, tile.buffers, tile.collision);\n        if (!bucket) continue;\n        bucket.features = [];\n        bucket.name = bucketName;\n        buckets[bucketName] = bucket;\n\n        if (data.layers) {\n            // vectortile\n            layerName = info['source-layer'];\n            if (!sourceLayers[layerName]) sourceLayers[layerName] = {};\n            sourceLayers[layerName][bucketName] = info;\n        } else {\n            // geojson tile\n            sourceLayers[bucketName] = info;\n        }\n    }\n\n    // read each layer, and sort its feature's into buckets\n    if (data.layers) {\n        // vectortile\n        for (layerName in sourceLayers) {\n            var layer = data.layers[layerName];\n            if (!layer) continue;\n            sortLayerIntoBuckets(layer, sourceLayers[layerName], buckets);\n        }\n    } else {\n        // geojson\n        sortLayerIntoBuckets(data, sourceLayers, buckets);\n    }\n\n    return buckets;\n}\n\n/*\n * Sorts features in a layer into different buckets, according to the maping\n *\n * Layers in vector tiles contain many different features, and feature types,\n * e.g. the landuse layer has parks, industrial buildings, forests, playgrounds\n * etc. However, when styling, we need to separate these features so that we can\n * render them separately with different styles.\n *\n * @param {VectorTileLayer} layer\n * @param {Mapping} mapping\n */\nfunction sortLayerIntoBuckets(layer, mapping, buckets) {\n    for (var i = 0; i < layer.length; i++) {\n        var feature = layer.feature(i);\n        for (var key in mapping) {\n            if (mapping[key].compare(feature)) {\n                buckets[key].features.push(feature);\n            }\n        }\n    }\n}\n\nfunction getGeometry(feature) {\n    return feature.loadGeometry();\n}\n\nfunction getType(feature) {\n    return vt.VectorTileFeature.types[feature.type];\n}\n","'use strict';\n\nmodule.exports = AnimationLoop;\n\nfunction AnimationLoop() {\n    this.n = 0;\n    this.times = [];\n}\n\n// Are all animations done?\nAnimationLoop.prototype.stopped = function() {\n    this.times = this.times.filter(function(t) {\n        return t.time >= (new Date()).getTime();\n    });\n    return !this.times.length;\n};\n\n// Add a new animation that will run t milliseconds\n// Returns an id that can be used to cancel it layer\nAnimationLoop.prototype.set = function(t) {\n    this.times.push({ id: this.n, time: t + (new Date()).getTime() });\n    return this.n++;\n};\n\n// Cancel an animation\nAnimationLoop.prototype.cancel = function(n) {\n    this.times = this.times.filter(function(t) {\n        return t.id != n;\n    });\n};\n","'use strict';\n\nvar VectorTileFeature = require('vector-tile').VectorTileFeature;\n\nfunction infix(operator) {\n    return function(left, right) { return left + ' ' + operator + ' ' + right; };\n}\n\nvar infixOperators = {\n    '==': infix('==='),\n    '>': infix('>'), '$gt': infix('>'),\n    '<': infix('<'), '$lt': infix('<'),\n    '<=': infix('<='), '$lte': infix('<='),\n    '>=': infix('>='), '$gte': infix('>='),\n    '!=': infix('!=='), '$ne': infix('!=='),\n    '$exists': function (value) { return value + ' !== undefined'; }\n};\n\nfunction or(items)  { return '(' + items.join(' || ') + ')'; }\nfunction and(items) { return '(' + items.join(' && ') + ')'; }\nfunction not(item)  { return '!' + item; }\nfunction nor(items) { return not(or(items)); }\n\nvar arrayOperators = {\n    '||': or, '$or': or,\n    '&&': and, '$and': and,\n    '!': nor, '$nor': nor\n};\n\nvar objOperators = {\n    '!': not, '$not': not\n};\n\nmodule.exports = function (filter) {\n    // simple key & value comparison\n    function valueFilter(key, value, operator) {\n        return operator('p[' + JSON.stringify(key) + ']', JSON.stringify(value));\n    }\n\n    // compares key & value or key & or(values)\n    function simpleFieldFilter(key, value, operator) {\n        var operatorFn = infixOperators[operator || '=='];\n        if (!operatorFn) throw new Error('Unknown operator: ' + operator);\n\n        if (Array.isArray(value)) {\n            return or(value.map(function (v) {\n                return valueFilter(key, v, operatorFn);\n            }));\n\n        } else return valueFilter(key, value, operatorFn);\n    }\n\n    // handles any filter key/value pair\n    function fieldFilter(key, value) {\n\n        if (Array.isArray(value)) {\n            if (key in arrayOperators) { // handle and/or operators\n                return arrayOperators[key](value.map(fieldsFilter));\n            }\n\n        } else if (typeof value === 'object') {\n\n            // handle not operator\n            if (key in objOperators) return objOperators[key](fieldsFilter(value));\n\n            // handle {key: {operator: value}} notation\n            var filters = [];\n            for (var op in value) {\n                filters.push(simpleFieldFilter(key, value[op], op));\n            }\n            return and(filters);\n\n        }\n        // handle simple key/value or key/values comparison\n        return simpleFieldFilter(key, value);\n    }\n\n    function typeFilter(type) {\n        return 'f.type === ' + VectorTileFeature.types.indexOf(type);\n    }\n\n    function fieldsFilter(obj) {\n        var filters = [];\n\n        for (var key in obj) {\n            if (key === '$type') {\n                filters.push(typeFilter(obj[key]));\n            } else {\n                filters.push(fieldFilter(key, obj[key]));\n            }\n        }\n\n        return filters.length ? and(filters) : 'true';\n    }\n\n    var filterStr = 'var p = f.properties || {}; return ' + fieldsFilter(filter || {}) + ';';\n\n    // jshint evil: true\n    return new Function('f', filterStr);\n};\n","'use strict';\n\nvar reference = require('mapbox-gl-style-spec/reference/v4');\n\nmodule.exports = {};\n\nreference['class'].forEach(function(className) {\n    var Calculated = function() {};\n    var style = reference[className];\n    for (var prop in style) {\n        if (style[prop]['default'] === undefined) continue;\n        Calculated.prototype[prop] = style[prop]['default'];\n    }\n    module.exports[className.replace('class_','')] = Calculated;\n});\n\n","'use strict';\n\nvar Evented = require('../util/evented.js');\nvar ajax = require('../util/ajax.js');\nvar browser = require('../util/browser.js');\n\nmodule.exports = ImageSprite;\n\nfunction ImageSprite(base) {\n\n    var sprite = this;\n    this.base = base;\n    this.retina = browser.devicePixelRatio > 1;\n\n    base = sprite.base + (sprite.retina ? '@2x' : '');\n\n    ajax.getJSON(base + '.json', function(err, data) {\n        // @TODO handle errors via sprite event.\n        if (err) return;\n        sprite.data = data;\n        if (sprite.img) sprite.fire('loaded');\n    });\n\n    ajax.getImage(base + '.png', function(err, img) {\n        // @TODO handle errors via sprite event.\n        if (err) return;\n\n        // premultiply the sprite\n        var data = img.getData();\n        var newdata = img.data = new Uint8Array(data.length);\n        for (var i = 0; i < data.length; i+=4) {\n            var alpha = data[i + 3] / 255;\n            newdata[i + 0] = data[i + 0] * alpha;\n            newdata[i + 1] = data[i + 1] * alpha;\n            newdata[i + 2] = data[i + 2] * alpha;\n            newdata[i + 3] = data[i + 3];\n        }\n\n        sprite.img = img;\n        if (sprite.data) sprite.fire('loaded');\n    });\n}\n\nImageSprite.prototype = Object.create(Evented);\n\nImageSprite.prototype.toJSON = function() {\n    return this.base;\n};\n\nImageSprite.prototype.loaded = function() {\n    return !!(this.data && this.img);\n};\n\nImageSprite.prototype.resize = function(gl) {\n    var sprite = this;\n    if (browser.devicePixelRatio > 1 !== sprite.retina) {\n\n        var newSprite = new ImageSprite(sprite.base);\n        newSprite.on('loaded', function() {\n\n            sprite.img = newSprite.img;\n            sprite.data = newSprite.data;\n            sprite.retina = newSprite.retina;\n\n            if (sprite.texture) {\n                gl.deleteTexture(sprite.texture);\n                delete sprite.texture;\n            }\n\n        });\n    }\n};\n\nImageSprite.prototype.bind = function(gl, linear) {\n    var sprite = this;\n\n    if (!sprite.loaded())\n        return;\n\n    if (!sprite.texture) {\n        sprite.texture = gl.createTexture();\n        gl.bindTexture(gl.TEXTURE_2D, sprite.texture);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n        var img = sprite.img;\n        gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, img.width, img.height, 0, gl.RGBA, gl.UNSIGNED_BYTE, img.data);\n\n    } else {\n        gl.bindTexture(gl.TEXTURE_2D, sprite.texture);\n    }\n\n    var filter = linear ? gl.LINEAR : gl.NEAREST;\n    if (filter !== sprite.filter) {\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, filter);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, filter);\n    }\n};\n\nImageSprite.prototype.getPosition = function(name, repeating) {\n\n    // `repeating` indicates that the image will be used in a repeating pattern\n    // repeating pattern images are assumed to have a 1px padding that mirrors the opposite edge\n    // positions for repeating images are adjusted to exclude the edge\n    repeating = repeating === true ? 1 : 0;\n\n    var pos = this.data && this.data[name];\n    if (pos && this.img) {\n        var width = this.img.width;\n        var height = this.img.height;\n        return {\n            size: [pos.width / pos.pixelRatio, pos.height / pos.pixelRatio],\n            tl: [(pos.x + repeating)/ width, (pos.y + repeating) / height],\n            br: [(pos.x + pos.width - 2 * repeating) / width, (pos.y + pos.height - 2 * repeating) / height]\n        };\n    }\n};\n","'use strict';\n\nvar reference = require('mapbox-gl-style-spec/reference/v4');\n\nmodule.exports = {};\n\nreference.render.forEach(function(className) {\n    var Properties = function(props) {\n        for (var p in props) {\n            this[p] = props[p];\n        }\n    };\n\n    var properties = reference[className];\n    for (var prop in properties) {\n        if (properties[prop]['default'] === undefined) continue;\n        Properties.prototype[prop] = properties[prop]['default'];\n    }\n    module.exports[className.replace('render_','')] = Properties;\n});\n\n","'use strict';\n\nvar Evented = require('../util/evented.js');\n\nvar StyleTransition = require('./styletransition.js');\nvar StyleDeclaration = require('./styledeclaration.js');\nvar StyleConstant = require('./styleconstant.js');\nvar CalculatedStyle = require('./calculatedstyle.js');\nvar ImageSprite = require('./imagesprite.js');\n\nvar util = require('../util/util.js');\n\nmodule.exports = Style;\n\n/*\n * The map style's current state\n *\n * The stylesheet object is not modified. To change the style, just change\n * the the stylesheet object and trigger a cascade.\n */\nfunction Style(stylesheet, animationLoop) {\n    if (stylesheet.version !== 4) console.warn('Stylesheet version must be 4');\n    if (!Array.isArray(stylesheet.layers)) console.warn('Stylesheet must have layers');\n\n    this.classes = {};\n    this.stylesheet = stylesheet;\n    this.animationLoop = animationLoop;\n\n    this.buckets = {};\n    this.orderedBuckets = [];\n    this.transitions = {};\n    this.computed = {};\n    this.sources = {};\n\n    this.cascade({transition: false});\n\n    if (stylesheet.sprite) this.setSprite(stylesheet.sprite);\n}\n\nStyle.prototype = Object.create(Evented);\n\nfunction premultiplyLayer(layer, type) {\n    var colorProp = type + '-color',\n        haloProp = type + '-halo-color',\n        outlineProp = type + '-outline-color',\n        color = layer[colorProp],\n        haloColor = layer[haloProp],\n        outlineColor = layer[outlineProp],\n        opacity = layer[type + '-opacity'];\n\n    var colorOpacity = color && (opacity * color[3]);\n    var haloOpacity = haloColor && (opacity * haloColor[3]);\n    var outlineOpacity = outlineColor && (opacity * outlineColor[3]);\n\n    if (colorOpacity !== undefined && colorOpacity < 1) {\n        layer[colorProp] = util.premultiply([color[0], color[1], color[2], colorOpacity]);\n    }\n    if (haloOpacity !== undefined && haloOpacity < 1) {\n        layer[haloProp] = util.premultiply([haloColor[0], haloColor[1], haloColor[2], haloOpacity]);\n    }\n    if (outlineOpacity !== undefined && outlineOpacity < 1) {\n        layer[outlineProp] = util.premultiply([outlineColor[0], outlineColor[1], outlineColor[2], outlineOpacity]);\n    }\n}\n\n// Formerly known as zoomed styles\nStyle.prototype.recalculate = function(z) {\n    if (typeof z !== 'number') console.warn('recalculate expects zoom level');\n\n    var transitions = this.transitions;\n    var layerValues = {};\n\n    this.sources = {};\n\n    this.rasterFadeDuration = 300;\n\n    for (var name in transitions) {\n        var layer = transitions[name],\n            bucket = this.buckets[layer.ref || name],\n            layerType = this.layermap[name].type;\n\n        if (!CalculatedStyle[layerType]) {\n            console.warn('unknown layer type ' + layerType);\n            continue;\n        }\n        var appliedLayer = layerValues[name] = new CalculatedStyle[layerType]();\n        for (var rule in layer) {\n            var transition = layer[rule];\n            appliedLayer[rule] = transition.at(z);\n        }\n\n        if (layerType === 'symbol') {\n            appliedLayer.hidden =\n                (appliedLayer['text-opacity'] === 0 || !bucket.render['text-field']) &&\n                (appliedLayer['icon-opacity'] === 0 || !bucket.render['icon-image']);\n            if (!appliedLayer.hidden) {\n                premultiplyLayer(appliedLayer, 'text');\n                premultiplyLayer(appliedLayer, 'icon');\n            }\n        } else {\n            appliedLayer.hidden = (appliedLayer[layerType + '-opacity'] === 0);\n            if (!appliedLayer.hidden) {\n                premultiplyLayer(appliedLayer, layerType);\n            }\n        }\n\n        // Find all the sources that are currently being used\n        // so that we can automatically enable/disable them as needed\n        if (!appliedLayer.hidden) {\n            var source = bucket && bucket.source;\n\n            // mark source as used so that tiles are downloaded\n            if (source) this.sources[source] = true;\n        }\n\n        if (appliedLayer['raster-fade']) {\n            this.rasterFadeDuration = Math.max(this.rasterFadeDuration, appliedLayer['raster-fade']);\n        }\n    }\n\n    this.computed = layerValues;\n\n    this.z = z;\n    this.fire('zoom');\n};\n\nStyle.prototype._simpleLayer = function(layer) {\n    var simple = {};\n    simple.id = layer.id;\n\n    var bucket = this.buckets[layer.ref || layer.id];\n    if (bucket) simple.bucket = bucket.id;\n    if (layer.type) simple.type = layer.type;\n\n    if (layer.layers) {\n        simple.layers = [];\n        for (var i = 0; i < layer.layers.length; i++) {\n            simple.layers.push(this._simpleLayer(layer.layers[i]));\n        }\n    }\n    return simple;\n};\n\n// Split the layers into groups of consecutive layers with the same datasource\n// For each group calculate its dependencies. Its dependencies are composited\n// layers that need to be rendered into textures before\nStyle.prototype._groupLayers = function(layers) {\n    var g = 0;\n    var groups = [];\n    var group;\n\n    // loop over layers top down\n    for (var i = layers.length - 1; i >= 0; i--) {\n        var layer = layers[i];\n\n        var bucket = this.buckets[layer.ref || layer.id];\n        var source = bucket && bucket.source;\n\n        // if the current layer is in a different source\n        if (group && source !== group.source) g++;\n\n        if (!groups[g]) {\n            group = [];\n            group.source = source;\n            if (layer.layers) group.composited = true;\n            groups[g] = group;\n        }\n\n        if (layer.layers && layer.type == 'composite') {\n            // TODO if composited layer is opaque just inline the layers\n            group.dependencies = group.dependencies || {};\n            group.dependencies[layer.id] = this._groupLayers(layer.layers);\n        }\n\n        group.push(this._simpleLayer(layer));\n    }\n\n    return groups;\n};\n\n/*\n * Take all the rules and declarations from the stylesheet,\n * and figure out which apply currently\n */\nStyle.prototype.cascade = function(options) {\n    options = options || {\n        transition: true\n    };\n\n    var a, b;\n    var id;\n    var prop;\n    var layer;\n    var className;\n    var styleName;\n    var style;\n    var styleTrans;\n    var constants = this.stylesheet.constants;\n\n    // derive buckets from layers\n    this.orderedBuckets = [];\n    this.buckets = getbuckets({}, this.orderedBuckets, this.stylesheet.layers);\n    function getbuckets(buckets, ordered, layers) {\n        for (var a = 0; a < layers.length; a++) {\n            var layer = layers[a];\n            if (layer.layers) {\n                buckets = getbuckets(buckets, ordered, layer.layers);\n            }\n            if (!layer.source || !layer.type) {\n                continue;\n            }\n            var bucket = { id: layer.id };\n            for (var prop in layer) {\n                if ((/^style/).test(prop)) continue;\n                bucket[prop] = layer[prop];\n            }\n            bucket.render = StyleConstant.resolve(bucket.render, constants);\n            buckets[layer.id] = bucket;\n            ordered.push(bucket);\n        }\n        return buckets;\n    }\n\n    // style class keys\n    var styleNames = ['style'];\n    for (className in this.classes) styleNames.push('style.' + className);\n\n    // apply layer group inheritance resulting in a flattened array\n    var flattened = flattenLayers(this.stylesheet.layers);\n\n    // map layer ids to layer definitions for resolving refs\n    var layermap = this.layermap = {};\n    for (a = 0; a < flattened.length; a++) {\n        layer = flattened[a];\n\n        var newLayer = {};\n        for (var k in layer) {\n            if (k === 'layers') continue;\n            newLayer[k] = layer[k];\n        }\n\n        layermap[layer.id] = newLayer;\n        flattened[a] = newLayer;\n    }\n\n    for (a = 0; a < flattened.length; a++) {\n        flattened[a] = resolveLayer(layermap, flattened[a]);\n    }\n\n    // Resolve layer references.\n    function resolveLayer(layermap, layer) {\n        if (!layer.ref || !layermap[layer.ref]) return layer;\n\n        var parent = resolveLayer(layermap, layermap[layer.ref]);\n        layer.render = parent.render;\n        layer.type = parent.type;\n        layer.filter = parent.filter;\n        layer.source = parent.source;\n        layer['source-layer'] = parent['source-layer'];\n        layer['min-zoom'] = parent['min-zoom'];\n        layer['max-zoom'] = parent['max-zoom'];\n\n        return layer;\n    }\n\n    // Flatten composite layer structures.\n    function flattenLayers(layers) {\n        var flat = [];\n        for (var i = 0; i < layers.length; i++) {\n            flat.push(layers[i]);\n            if (layers[i].layers) {\n                flat.push.apply(flat, flattenLayers(layers[i].layers));\n            }\n        }\n        return flat;\n    }\n\n    var transitions = {};\n    var globalTrans = this.stylesheet.transition;\n\n    for (a in flattened) {\n        layer = flattened[a];\n\n        id = layer.id;\n        style = {};\n        styleTrans = {};\n\n        // basic cascading of styles\n        for (b = 0; b < styleNames.length; b++) {\n            styleName = styleNames[b];\n            if (!layer[styleName]) continue;\n            // set style properties\n            for (prop in layer[styleName]) {\n                if (prop.indexOf('transition-') === -1) {\n                    style[prop] = layer[styleName][prop];\n                } else {\n                    styleTrans[prop.replace('transition-', '')] = layer[styleName][prop];\n                }\n            }\n        }\n\n        style = StyleConstant.resolve(style, constants);\n\n        var renderType = layer.type;\n        transitions[id] = {};\n\n        for (prop in style) {\n            var newDeclaration = new StyleDeclaration(renderType, prop, style[prop]);\n            var oldTransition = this.transitions[id] && this.transitions[id][prop];\n            var newStyleTrans = {};\n            newStyleTrans.duration = styleTrans[prop] && styleTrans[prop].duration ? styleTrans[prop].duration : globalTrans && globalTrans.duration ? globalTrans.duration : 300;\n            newStyleTrans.delay = styleTrans[prop] && styleTrans[prop].delay ? styleTrans[prop].delay : globalTrans && globalTrans.delay ? globalTrans.delay : 0;\n\n            if (!options.transition) {\n                newStyleTrans.duration = 0;\n                newStyleTrans.delay = 0;\n            }\n\n            // Only create a new transition if the declaration changed\n            if (!oldTransition || oldTransition.declaration.json !== newDeclaration.json) {\n                var newTransition = new StyleTransition(newDeclaration, oldTransition, newStyleTrans);\n                transitions[id][prop] = newTransition;\n\n                // Run the animation loop until the end of the transition\n                if (!newTransition.instant()) {\n                    newTransition.loopID = this.animationLoop.set(newTransition.endTime - (new Date()).getTime());\n                }\n\n                if (oldTransition) {\n                    this.animationLoop.cancel(oldTransition.loopID);\n                }\n            } else {\n                transitions[id][prop] = oldTransition;\n            }\n        }\n    }\n\n    this.transitions = transitions;\n    this.layerGroups = this._groupLayers(this.stylesheet.layers);\n\n    this.fire('change');\n};\n\n/* This should be moved elsewhere. Localizing resources doesn't belong here */\nStyle.prototype.setSprite = function(sprite) {\n    this.sprite = new ImageSprite(sprite);\n    this.sprite.on('loaded', this.fire.bind(this, 'change'));\n};\n\n// Modify classes\nStyle.prototype.addClass = function(n, options) {\n    if (this.classes[n]) return; // prevent unnecessary recalculation\n    this.classes[n] = true;\n    this.cascade(options);\n};\n\nStyle.prototype.removeClass = function(n, options) {\n    if (!this.classes[n]) return; // prevent unnecessary recalculation\n    delete this.classes[n];\n    this.cascade(options);\n};\n\nStyle.prototype.hasClass = function(n) {\n    return !!this.classes[n];\n};\n\nStyle.prototype.setClassList = function(l, options) {\n    this.classes = {};\n    for (var i = 0; i < l.length; i++) {\n        this.classes[l[i]] = true;\n    }\n    this.cascade(options);\n};\n\nStyle.prototype.getClassList = function() {\n    return Object.keys(this.classes);\n};\n\nStyle.prototype.getLayer = function(id) {\n    return this.layermap[id];\n};\n","'use strict';\n\nvar util = require('../util/util.js');\n\nmodule.exports.resolve = function (properties, constants) {\n    if (!constants)\n        return properties;\n\n    var result = {}, i;\n\n    function resolve(value) {\n        return typeof value === 'string' && value[0] === '@' ? constants[value] : value;\n    }\n\n    for (var key in properties) {\n        var value = resolve(properties[key]);\n\n        if (Array.isArray(value)) {\n            value = value.slice();\n\n            for (i = 0; i < value.length; i++) {\n                if (value[i] in constants) {\n                    value[i] = resolve(value[i]);\n                }\n            }\n        }\n\n        if (value.stops) {\n            value = util.extend({}, value);\n            value.stops = value.stops.slice();\n\n            for (i = 0; i < value.stops.length; i++) {\n                if (value.stops[i][1] in constants) {\n                    value.stops[i] = [\n                                value.stops[i][0],\n                        resolve(value.stops[i][1])\n                    ];\n                }\n            }\n        }\n\n        result[key] = value;\n    }\n\n    return result;\n};\n","'use strict';\n\nvar util = require('../util/util.js'),\n    reference = require('mapbox-gl-style-spec/reference/v4'),\n    parseCSSColor = require('csscolorparser').parseCSSColor;\n\nmodule.exports = StyleDeclaration;\n\n/*\n * A parsed representation of a property:value pair\n */\nfunction StyleDeclaration(renderType, prop, value) {\n    var className = 'class_' + renderType;\n    var propReference = reference[className] && reference[className][prop];\n    if (!propReference) return;\n\n    this.value = this.parseValue(value, propReference.type, propReference.values);\n    this.prop = prop;\n    this.type = propReference.type;\n\n    // immuatable representation of value. used for comparison\n    this.json = JSON.stringify(value);\n\n}\n\nStyleDeclaration.prototype.calculate = function(z) {\n    return typeof this.value === 'function' ? this.value(z) : this.value;\n};\n\nStyleDeclaration.prototype.parseValue = function(value, type, values) {\n    if (type === 'color') {\n        return parseColor(value);\n    } else if (type === 'number') {\n        return parseNumber(value);\n    } else if (type === 'boolean') {\n        return Boolean(value);\n    } else if (type === 'image') {\n        return String(value);\n    } else if (type === 'string') {\n        return String(value);\n    } else if (type === 'array') {\n        return parseNumberArray(value);\n    } else if (type === 'enum' && Array.isArray(values)) {\n        return values.indexOf(value) >= 0 ? value : undefined;\n    } else {\n        console.warn(type + ' is not a supported property type');\n    }\n};\n\nfunction parseNumber(num) {\n    if (num.stops) num = stopsFn(num);\n    var value = +num;\n    return !isNaN(value) ? value : num;\n}\n\nfunction parseNumberArray(array) {\n    var widths = array.map(parseNumber);\n\n    return function(z) {\n        var result = [];\n        for (var i = 0; i < widths.length; i++) {\n            result.push(typeof widths[i] === 'function' ? widths[i](z) : widths[i]);\n        }\n        return result;\n    };\n}\n\nvar colorCache = {};\n\nfunction parseColor(value) {\n    if (value.stops) {\n        for (var i = 0; i < value.stops.length; i++) {\n            // store the parsed color as the 3rd element in the array\n            value.stops[i][2] = parseCSSColor(value.stops[i][1]);\n        }\n        return stopsFn(value, true);\n    }\n\n    if (colorCache[value]) {\n        return colorCache[value];\n    }\n    var color = prepareColor(parseCSSColor(value));\n    colorCache[value] = color;\n    return color;\n}\n\nfunction stopsFn(params, color) {\n    var stops = params.stops;\n    var base = params.base || reference.function.base.default;\n\n    return function(z) {\n\n        // find the two stops which the current z is between\n        var low = null;\n        var high = null;\n        for (var i = 0; i < stops.length; i++) {\n            var stop = stops[i];\n            if (stop[0] <= z) low = stop;\n            if (stop[0] > z) {\n                high = stop;\n                break;\n            }\n        }\n\n        if (low && high) {\n            var zoomDiff = high[0] - low[0];\n            var zoomProgress = z - low[0];\n            var t = 0;\n            if (base == 1) {\n                t = zoomProgress / zoomDiff;\n            } else {\n                t = (Math.pow(base, zoomProgress) - 1) / (Math.pow(base, zoomDiff) - 1);\n            }\n            if (color) return prepareColor(interpColor(low[2], high[2], t));\n            else return util.interp(low[1], high[1], t);\n\n        } else if (low) {\n            if (color) return prepareColor(low[2]);\n            else return low[1];\n\n        } else if (high) {\n            if (color) return prepareColor(high[2]);\n            else return high[1];\n\n        } else {\n            if (color) return [0, 0, 0, 1];\n            else return 1;\n        }\n    };\n}\n\nfunction prepareColor(c) {\n    return [c[0] / 255, c[1] / 255, c[2] / 255, c[3] / 1];\n}\n\nfunction interpColor(from, to, t) {\n    return [\n        util.interp(from[0], to[0], t),\n        util.interp(from[1], to[1], t),\n        util.interp(from[2], to[2], t),\n        util.interp(from[3], to[3], t)\n    ];\n}\n","'use strict';\n\nvar util = require('../util/util.js');\n\nmodule.exports = StyleTransition;\n\n/*\n * Represents a transition between two declarations\n */\nfunction StyleTransition(declaration, oldTransition, value) {\n\n    this.declaration = declaration;\n    this.startTime = this.endTime = (new Date()).getTime();\n\n    var type = declaration.type;\n    if (type === 'number') {\n        this.interp = util.interp;\n    } else if (type === 'color') {\n        this.interp = interpColor;\n    } else if (type === 'array') {\n        this.interp = interpNumberArray;\n    }\n\n    this.oldTransition = oldTransition;\n    this.duration = value.duration || 0;\n    this.delay = value.delay || 0;\n\n    if (!this.instant()) {\n        this.endTime = this.startTime + this.duration + this.delay;\n        this.ease = util.easeCubicInOut;\n    }\n\n    if (oldTransition && oldTransition.endTime <= this.startTime) {\n        // Old transition is done running, so we can\n        // delete its reference to its old transition.\n\n        delete oldTransition.oldTransition;\n    }\n}\n\nStyleTransition.prototype.instant = function() {\n    return !this.oldTransition || !this.interp || (this.duration === 0 && this.delay === 0);\n};\n\n/*\n * Return the value of the transitioning property at zoom level `z` and optional time `t`\n */\nStyleTransition.prototype.at = function(z, t) {\n\n    var value = this.declaration.calculate(z);\n\n    if (this.instant()) return value;\n\n    t = t || Date.now();\n\n    if (t < this.endTime) {\n        var oldValue = this.oldTransition.at(z, this.startTime);\n        var eased = this.ease((t - this.startTime - this.delay) / this.duration);\n        value = this.interp(oldValue, value, eased);\n    }\n\n    return value;\n\n};\n\nfunction interpNumberArray(from, to, t) {\n    return from.map(function(d, i) {\n        return util.interp(d, to[i], t);\n    });\n}\n\nfunction interpColor(from, to, t) {\n    return [\n        util.interp(from[0], to[0], t),\n        util.interp(from[1], to[1], t),\n        util.interp(from[2], to[2], t),\n        util.interp(from[3], to[3], t)\n    ];\n}\n","'use strict';\n\nvar Point = require('point-geometry');\n\nmodule.exports = Anchor;\n\nfunction Anchor(x, y, angle, scale, segment) {\n    this.x = x;\n    this.y = y;\n    this.angle = angle;\n    this.scale = scale;\n\n    if (segment !== undefined) {\n        this.segment = segment;\n    }\n}\n\nAnchor.prototype = Object.create(Point.prototype);\n\nAnchor.prototype.clone = function() {\n    return new Anchor(this.x, this.y, this.angle, this.scale, this.segment);\n};\n","'use strict';\n\nmodule.exports = BinPack;\nfunction BinPack(width, height) {\n    this.width = width;\n    this.height = height;\n    this.free = [{ x: 0, y: 0, w: width, h: height }];\n}\n\nBinPack.prototype.release = function(rect) {\n    // Simple algorithm to recursively merge the newly released cell with its\n    // neighbor. This doesn't merge more than two cells at a time, and fails\n    // for complicated merges.\n    for (var i = 0; i < this.free.length; i++) {\n        var free = this.free[i];\n        if (free.y == rect.y && free.h == rect.h && free.x + free.w == rect.x) {\n            free.w += rect.w;\n        }\n        else if (free.x == rect.x && free.w == rect.w && free.y + free.h == rect.y) {\n            free.h += rect.h;\n        }\n        else if (rect.y == free.y && rect.h == free.h && rect.x + rect.w == free.x) {\n            free.x = rect.x;\n            free.w += rect.w;\n        }\n        else if (rect.x == free.x && rect.w == free.w && rect.y + rect.h == free.y) {\n            free.y = rect.y;\n            free.h += rect.h;\n        } else {\n            continue;\n        }\n\n        this.free.splice(i, 1);\n        this.release(free);\n        return;\n\n    }\n    this.free.push(rect);\n};\n\nBinPack.prototype.allocate = function(width, height) {\n    // Find the smallest free rect angle\n    var rect = { x: Infinity, y: Infinity, w: Infinity, h: Infinity };\n    var smallest = -1;\n    for (var i = 0; i < this.free.length; i++) {\n        var ref = this.free[i];\n        if (width <= ref.w && height <= ref.h && ref.y <= rect.y && ref.x <= rect.x) {\n            rect = ref;\n            smallest = i;\n        }\n    }\n\n    if (smallest < 0) {\n        // There's no space left for this char.\n        return { x: -1, y: -1 };\n    } else {\n        this.free.splice(smallest, 1);\n\n        // Shorter/Longer Axis Split Rule (SAS)\n        // http://clb.demon.fi/files/RectangleBinPack.pdf p. 15\n        // Ignore the dimension of R and just split long the shorter dimension\n        // See Also: http://www.cs.princeton.edu/~chazelle/pubs/blbinpacking.pdf\n        if (rect.w < rect.h) {\n            // split horizontally\n            // +--+---+\n            // |__|___|  <-- b1\n            // +------+  <-- b2\n            if (rect.w > width) this.free.push({ x: rect.x + width, y: rect.y, w: rect.w - width, h: height });\n            if (rect.h > height) this.free.push({ x: rect.x, y: rect.y + height, w: rect.w, h: rect.h - height });\n        } else {\n            // split vertically\n            // +--+---+\n            // |__|   | <-- b1\n            // +--|---+ <-- b2\n            if (rect.w > width) this.free.push({ x: rect.x + width, y: rect.y, w: rect.w - width, h: rect.h });\n            if (rect.h > height) this.free.push({ x: rect.x, y: rect.y + height, w: width, h: rect.h - height });\n        }\n\n        return { x: rect.x, y: rect.y, w: width, h: height };\n    }\n};\n","'use strict';\n\nvar rbush = require('rbush'),\n    rotationRange = require('./rotationrange.js'),\n    Point = require('point-geometry');\n\nmodule.exports = Collision;\n\nfunction Collision(zoom, tileExtent, tileSize, placementDepth) {\n    this.hTree = rbush(); // tree for horizontal labels\n    this.cTree = rbush(); // tree for glyphs from curved labels\n\n    // tile pixels per screen pixels at the tile's zoom level\n    this.tilePixelRatio = tileExtent / tileSize;\n\n    this.zoom = zoom;\n\n    // Calculate the maximum scale we can go down in our fake-3d rtree so that\n    // placement still makes sense. This is calculated so that the minimum\n    // placement zoom can be at most 25.5 (we use an unsigned integer x10 to\n    // store the minimum zoom).\n    //\n    // We don't want to place labels all the way to 25.5. This lets too many\n    // glyphs be placed, slowing down collision checking. Only place labels if\n    // they will show up within the intended zoom range of the tile.\n    placementDepth = Math.min(3, placementDepth || 1, 25.5 - this.zoom);\n    this.maxPlacementScale = Math.exp(Math.LN2 * placementDepth);\n\n    var m = 4096;\n    var edge = m * this.tilePixelRatio * 2;\n\n    var fullRange = [Math.PI * 2, 0];\n\n    this.left = {\n        anchor: new Point(0, 0),\n        box: { x1: -edge, y1: -edge, x2: 0, y2: edge },\n        placementRange: fullRange,\n        placementScale: 0.5,\n        maxScale: Infinity,\n        padding: 0\n    };\n\n    this.top = {\n        anchor: new Point(0, 0),\n        box: { x1: -edge, y1: -edge, x2: edge, y2: 0 },\n        placementRange: fullRange,\n        placementScale: 0.5,\n        maxScale: Infinity,\n        padding: 0\n    };\n\n    this.bottom = {\n        anchor: new Point(m, m),\n        box: { x1: -edge, y1: 0, x2: edge, y2: edge },\n        placementRange: fullRange,\n        placementScale: 0.5,\n        maxScale: Infinity,\n        padding: 0\n    };\n\n    this.right = {\n        anchor: new Point(m, m),\n        box: { x1: 0, y1: -edge, x2: edge, y2: edge },\n        placementRange: fullRange,\n        placementScale: 0.5,\n        maxScale: Infinity,\n        padding: 0\n    };\n\n}\n\nCollision.prototype.getPlacementScale = function(glyphs, minPlacementScale, avoidEdges) {\n\n    var left = this.left;\n    var right = this.right;\n    var top = this.top;\n    var bottom = this.bottom;\n\n    for (var k = 0; k < glyphs.length; k++) {\n\n        var glyph = glyphs[k];\n        var box = glyph.box;\n        var bbox = glyph.hBox || box;\n        var anchor = glyph.anchor;\n        var pad = glyph.padding;\n\n        var minScale = Math.max(minPlacementScale, glyph.minScale);\n        var maxScale = glyph.maxScale || Infinity;\n\n        if (minScale >= maxScale) continue;\n\n        // Compute the scaled bounding box of the unrotated glyph\n        var searchBox = this.getBox(anchor, bbox, minScale, maxScale);\n\n        var blocking = this.hTree.search(searchBox).concat(this.cTree.search(searchBox));\n\n        if (avoidEdges) {\n            if (searchBox[0] < 0) blocking.push(left);\n            if (searchBox[1] < 0) blocking.push(top);\n            if (searchBox[2] >= 4096) blocking.push(right);\n            if (searchBox[3] >= 4096) blocking.push(bottom);\n        }\n\n        if (blocking.length) {\n\n            var na = anchor; // new anchor\n            var nb = box; // new box\n\n            for (var l = 0; l < blocking.length; l++) {\n                var oa = blocking[l].anchor; // old anchor\n                var ob = blocking[l].box; // old box\n\n                // If anchors are identical, we're going to skip the label.\n                // NOTE: this isn't right because there can be glyphs with\n                // the same anchor but differing box offsets.\n                if (na.equals(oa)) {\n                    return null;\n                }\n\n                // todo: unhardcode the 8 = tileExtent/tileSize\n                var padding = Math.max(pad, blocking[l].padding) * 8;\n\n                // Original algorithm:\n                var s1 = (ob.x1 - nb.x2 - padding) / (na.x - oa.x); // scale at which new box is to the left of old box\n                var s2 = (ob.x2 - nb.x1 + padding) / (na.x - oa.x); // scale at which new box is to the right of old box\n                var s3 = (ob.y1 - nb.y2 - padding) / (na.y - oa.y); // scale at which new box is to the top of old box\n                var s4 = (ob.y2 - nb.y1 + padding) / (na.y - oa.y); // scale at which new box is to the bottom of old box\n\n                if (isNaN(s1) || isNaN(s2)) s1 = s2 = 1;\n                if (isNaN(s3) || isNaN(s4)) s3 = s4 = 1;\n\n                var collisionFreeScale = Math.min(Math.max(s1, s2), Math.max(s3, s4));\n\n                // Only update label's min scale if the glyph was restricted by a collision\n                if (collisionFreeScale > minPlacementScale &&\n                    collisionFreeScale > minScale &&\n                    collisionFreeScale < maxScale &&\n                    collisionFreeScale < blocking[l].maxScale) {\n                    minPlacementScale = collisionFreeScale;\n                }\n\n                if (minPlacementScale > this.maxPlacementScale) {\n                    return null;\n                }\n            }\n\n        }\n    }\n\n    return minPlacementScale;\n};\n\nCollision.prototype.getPlacementRange = function(glyphs, placementScale, horizontal) {\n\n    var placementRange = [2*Math.PI, 0];\n\n    for (var k = 0; k < glyphs.length; k++) {\n        var glyph = glyphs[k];\n        var bbox = glyph.hBox || glyph.box;\n        var anchor = glyph.anchor;\n\n        var minPlacedX = anchor.x + bbox.x1 / placementScale;\n        var minPlacedY = anchor.y + bbox.y1 / placementScale;\n        var maxPlacedX = anchor.x + bbox.x2 / placementScale;\n        var maxPlacedY = anchor.y + bbox.y2 / placementScale;\n\n        var searchBox = [minPlacedX, minPlacedY, maxPlacedX, maxPlacedY];\n\n        var blocking = this.hTree.search(searchBox);\n\n        if (horizontal) {\n            blocking = blocking.concat(this.cTree.search(searchBox));\n        }\n\n        for (var l = 0; l < blocking.length; l++) {\n            var b = blocking[l];\n            var bbox2 = b.hBox || b.box;\n\n            var x1, x2, y1, y2, intersectX, intersectY;\n\n            // Adjust and compare bboxes to see if the glyphs might intersect\n            if (placementScale > b.placementScale) {\n                x1 = b.anchor.x + bbox2.x1 / placementScale;\n                y1 = b.anchor.y + bbox2.y1 / placementScale;\n                x2 = b.anchor.x + bbox2.x2 / placementScale;\n                y2 = b.anchor.y + bbox2.y2 / placementScale;\n                intersectX = x1 < maxPlacedX && x2 > minPlacedX;\n                intersectY = y1 < maxPlacedY && y2 > minPlacedY;\n            } else {\n                x1 = anchor.x + bbox.x1 / b.placementScale;\n                y1 = anchor.y + bbox.y1 / b.placementScale;\n                x2 = anchor.x + bbox.x2 / b.placementScale;\n                y2 = anchor.y + bbox.y2 / b.placementScale;\n                intersectX = x1 < b[2] && x2 > b[0];\n                intersectY = y1 < b[3] && y2 > b[1];\n            }\n\n            // If they can't intersect, skip more expensive rotation calculation\n            if (!(intersectX && intersectY)) continue;\n\n            var scale = Math.max(placementScale, b.placementScale);\n            var range = rotationRange.rotationRange(glyph, b, scale);\n\n            placementRange[0] = Math.min(placementRange[0], range[0]);\n            placementRange[1] = Math.max(placementRange[1], range[1]);\n        }\n    }\n\n    return placementRange;\n\n};\n\n// Insert glyph placements into rtree.\nCollision.prototype.insert = function(glyphs, anchor, placementScale, placementRange, horizontal) {\n\n    var allBounds = [];\n\n    for (var k = 0; k < glyphs.length; k++) {\n\n        var glyph = glyphs[k];\n        var bbox = glyph.hBox || glyph.box;\n\n        var minScale = Math.max(placementScale, glyph.minScale);\n        var maxScale = glyph.maxScale || Infinity;\n\n        var bounds = this.getBox(anchor, bbox, minScale, maxScale);\n\n        bounds.anchor = anchor;\n        bounds.box = glyph.box;\n        if (glyph.hBox) bounds.hBox = bbox;\n        bounds.placementRange = placementRange;\n        bounds.placementScale = minScale;\n        bounds.maxScale = maxScale;\n        bounds.padding = glyph.padding;\n\n        allBounds.push(bounds);\n    }\n\n    (horizontal ? this.hTree : this.cTree).load(allBounds);\n};\n\nCollision.prototype.getBox = function(anchor, bbox, minScale, maxScale) {\n    return [\n        anchor.x + Math.min(bbox.x1 / minScale, bbox.x1 / maxScale),\n        anchor.y + Math.min(bbox.y1 / minScale, bbox.y1 / maxScale),\n        anchor.x + Math.max(bbox.x2 / minScale, bbox.x2 / maxScale),\n        anchor.y + Math.max(bbox.y2 / minScale, bbox.y2 / maxScale)];\n};\n","'use strict';\n\nvar BinPack = require('./binpack.js');\n\nmodule.exports = GlyphAtlas;\nfunction GlyphAtlas(width, height) {\n    this.width = width;\n    this.height = height;\n\n    this.bin = new BinPack(width, height);\n    this.index = {};\n    this.ids = {};\n    this.data = new Uint8Array(width * height);\n}\n\nGlyphAtlas.prototype = {\n    get debug() {\n        return 'canvas' in this;\n    },\n    set debug(value) {\n        if (value && !this.canvas) {\n            this.canvas = document.createElement('canvas');\n            this.canvas.width = this.width;\n            this.canvas.height = this.height;\n            document.body.appendChild(this.canvas);\n            this.ctx = this.canvas.getContext('2d');\n        } else if (!value && this.canvas) {\n            this.canvas.parentNode.removeChild(this.canvas);\n            delete this.ctx;\n            delete this.canvas;\n        }\n    }\n};\n\nGlyphAtlas.prototype.getGlyphs = function() {\n    var glyphs = {},\n        split,\n        name,\n        id;\n\n    for (var key in this.ids) {\n        split = key.split('#');\n        name = split[0];\n        id = split[1];\n\n        if (!glyphs[name]) glyphs[name] = [];\n        glyphs[name].push(id);\n    }\n\n    return glyphs;\n};\n\nGlyphAtlas.prototype.getRects = function() {\n    var rects = {},\n        split,\n        name,\n        id;\n\n    for (var key in this.ids) {\n        split = key.split('#');\n        name = split[0];\n        id = split[1];\n\n        if (!rects[name]) rects[name] = {};\n        rects[name][id] = this.index[key];\n    }\n\n    return rects;\n};\n\nGlyphAtlas.prototype.removeGlyphs = function(id) {\n    for (var key in this.ids) {\n\n        var ids = this.ids[key];\n\n        var pos = ids.indexOf(id);\n        if (pos >= 0) ids.splice(pos, 1);\n        this.ids[key] = ids;\n\n        if (!ids.length) {\n            var rect = this.index[key];\n\n            var target = this.data;\n            for (var y = 0; y < rect.h; y++) {\n                var y1 = this.width * (rect.y + y) + rect.x;\n                for (var x = 0; x < rect.w; x++) {\n                    target[y1 + x] = 0;\n                }\n            }\n\n            this.dirty = true;\n\n            this.bin.release(rect);\n\n            delete this.index[key];\n            delete this.ids[key];\n        }\n    }\n\n\n    this.updateTexture(this.gl);\n};\n\nGlyphAtlas.prototype.addGlyph = function(id, name, glyph, buffer) {\n    if (!glyph) {\n        // console.warn('missing glyph', code, String.fromCharCode(code));\n        return null;\n    }\n    var key = name + \"#\" + glyph.id;\n\n    // The glyph is already in this texture.\n    if (this.index[key]) {\n        if (this.ids[key].indexOf(id) < 0) {\n            this.ids[key].push(id);\n        }\n        return this.index[key];\n    }\n\n    // The glyph bitmap has zero width.\n    if (!glyph.bitmap) {\n        return null;\n    }\n\n    var buffered_width = glyph.width + buffer * 2;\n    var buffered_height = glyph.height + buffer * 2;\n\n    // Add a 1px border around every image.\n    var pack_width = buffered_width;\n    var pack_height = buffered_height;\n\n    // Increase to next number divisible by 4, but at least 1.\n    // This is so we can scale down the texture coordinates and pack them\n    // into 2 bytes rather than 4 bytes.\n    pack_width += (4 - pack_width % 4);\n    pack_height += (4 - pack_height % 4);\n\n    var rect = this.bin.allocate(pack_width, pack_height);\n    if (rect.x < 0) {\n        console.warn('glyph bitmap overflow');\n        return { glyph: glyph, rect: null };\n    }\n\n    // Add left and top glyph offsets to rect.\n    rect.l = glyph.left;\n    rect.t = glyph.top;\n\n    this.index[key] = rect;\n    this.ids[key] = [id];\n\n    var target = this.data;\n    var source = glyph.bitmap;\n    for (var y = 0; y < buffered_height; y++) {\n        var y1 = this.width * (rect.y + y) + rect.x;\n        var y2 = buffered_width * y;\n        for (var x = 0; x < buffered_width; x++) {\n            target[y1 + x] = source[y2 + x];\n        }\n    }\n\n    this.dirty = true;\n\n    return rect;\n};\n\nGlyphAtlas.prototype.bind = function(gl) {\n    this.gl = gl;\n    if (!this.texture) {\n        this.texture = gl.createTexture();\n        gl.bindTexture(gl.TEXTURE_2D, this.texture);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n        gl.texImage2D(gl.TEXTURE_2D, 0, gl.ALPHA, this.width, this.height, 0, gl.ALPHA, gl.UNSIGNED_BYTE, null);\n\n    } else {\n        gl.bindTexture(gl.TEXTURE_2D, this.texture);\n    }\n};\n\nGlyphAtlas.prototype.updateTexture = function(gl) {\n    this.bind(gl);\n    if (this.dirty) {\n\n        gl.texSubImage2D(gl.TEXTURE_2D, 0, 0, 0, this.width, this.height, gl.ALPHA, gl.UNSIGNED_BYTE, this.data);\n\n        // DEBUG\n        if (this.ctx) {\n            var data = this.ctx.getImageData(0, 0, this.width, this.height);\n            for (var i = 0, j = 0; i < this.data.length; i++, j += 4) {\n                data.data[j] = this.data[i];\n                data.data[j+1] = this.data[i];\n                data.data[j+2] = this.data[i];\n                data.data[j+3] = 255;\n            }\n            this.ctx.putImageData(data, 0, 0);\n\n            this.ctx.strokeStyle = 'red';\n            for (var k = 0; k < this.bin.free.length; k++) {\n                var free = this.bin.free[k];\n                this.ctx.strokeRect(free.x, free.y, free.w, free.h);\n            }\n        }\n        // END DEBUG\n\n        this.dirty = false;\n    }\n};\n","'use strict';\n\nvar getArrayBuffer = require('../util/ajax.js').getArrayBuffer;\nvar Glyphs = require('../util/glyphs.js');\nvar Protobuf = require('pbf');\n\nmodule.exports = GlyphSource;\n\nfunction GlyphSource(url, glyphAtlas) {\n    this.url = url;\n    this.glyphAtlas = glyphAtlas;\n    this.stacks = {};\n    this.loading = {};\n}\n\nGlyphSource.prototype.getRects = function(fontstack, glyphIDs, tileID, callback) {\n\n    if (this.stacks[fontstack] === undefined) this.stacks[fontstack] = {};\n\n    var rects = {};\n    var glyphs = {};\n    var result = { rects: rects, glyphs: glyphs };\n\n    var stack = this.stacks[fontstack];\n    var glyphAtlas = this.glyphAtlas;\n\n    var missing = {};\n    var remaining = 0;\n\n    for (var i = 0; i < glyphIDs.length; i++) {\n        var glyphID = glyphIDs[i];\n        var range = Math.floor(glyphID / 256);\n\n        if (stack[range]) {\n            var glyph = stack[range].glyphs[glyphID];\n            var buffer = 3;\n            rects[glyphID] = glyphAtlas.addGlyph(tileID, fontstack, glyph, buffer);\n            if (glyph) glyphs[glyphID] = simpleGlyph(glyph);\n        } else {\n            if (missing[range] === undefined) {\n                missing[range] = [];\n                remaining++;\n            }\n            missing[range].push(glyphID);\n        }\n    }\n\n    if (!remaining) callback(undefined, result);\n\n    var glyphSource = this;\n    for (var r in missing) {\n        this.loadRange(fontstack, r, onRangeLoaded);\n    }\n\n    function onRangeLoaded(err, range, data) {\n        // TODO not be silent about errors\n        if (!err) {\n            var stack = glyphSource.stacks[fontstack][range] = data.stacks[fontstack];\n            for (var i = 0; i < missing[range].length; i++) {\n                var glyphID = missing[range][i];\n                var glyph = stack.glyphs[glyphID];\n                var buffer = 3;\n                rects[glyphID] = glyphAtlas.addGlyph(tileID, fontstack, glyph, buffer);\n                if (glyph) glyphs[glyphID] = simpleGlyph(glyph);\n            }\n        }\n        remaining--;\n        if (!remaining) callback(undefined, result);\n    }\n};\n\nfunction simpleGlyph(glyph) {\n    return {\n        advance: glyph.advance,\n        left: glyph.left,\n        top: glyph.top\n    };\n}\n\nGlyphSource.prototype.loadRange = function(fontstack, range, callback) {\n\n    if (range * 256 >= 65280) return callback('gyphs > 65280 not supported');\n\n    if (this.loading[fontstack] === undefined) this.loading[fontstack] = {};\n    var loading = this.loading[fontstack];\n\n    if (loading[range]) {\n        loading[range].push(callback);\n    } else {\n        loading[range] = [callback];\n\n        var rangeName = (range * 256) + '-' + (range * 256 + 255);\n        var url = glyphUrl(fontstack, rangeName, this.url);\n\n        getArrayBuffer(url, function(err, data) {\n            var glyphs = !err && new Glyphs(new Protobuf(new Uint8Array(data)));\n            for (var i = 0; i < loading[range].length; i++) {\n                loading[range][i](err, range, glyphs);\n            }\n            delete loading[range];\n        });\n    }\n};\n\nfunction glyphUrl(fontstack, range, url, subdomains) {\n    subdomains = subdomains || 'abc';\n\n    return url\n        .replace('{s}', subdomains[fontstack.length % subdomains.length])\n        .replace('{fontstack}', fontstack)\n        .replace('{range}', range);\n}\n","'use strict';\n\nvar util = require('../util/util.js'),\n    Anchor = require('../symbol/anchor.js');\n\nmodule.exports = interpolate;\n\nvar minScale = 0.5;\nvar minScaleArrays = {\n    1: [minScale],\n    2: [minScale, 2],\n    4: [minScale, 4, 2, 4],\n    8: [minScale, 8, 4, 8, 2, 8, 4, 8]\n};\n\n\nfunction interpolate(vertices, spacing, minScale, maxScale, tilePixelRatio, start) {\n\n    if (minScale === undefined) minScale = 0;\n\n    maxScale = Math.round(Math.max(Math.min(8, maxScale / 2), 1));\n    spacing *= tilePixelRatio / maxScale;\n    var minScales = minScaleArrays[maxScale];\n    var len = minScales.length;\n\n    var distance = 0,\n        markedDistance = 0,\n        added = start || 0;\n\n    var points = [];\n\n    for (var i = 0; i < vertices.length - 1; i++) {\n\n        var a = vertices[i],\n            b = vertices[i + 1];\n\n        var segmentDist = a.dist(b),\n            angle = b.angleTo(a);\n\n        while (markedDistance + spacing < distance + segmentDist) {\n            markedDistance += spacing;\n\n            var t = (markedDistance - distance) / segmentDist,\n                x = util.interp(a.x, b.x, t),\n                y = util.interp(a.y, b.y, t),\n                s = minScales[added % len];\n\n            if (x >= 0 && x < 4096 && y >= 0 && y < 4096) {\n                points.push(new Anchor(x, y, angle, s, i));\n            }\n\n            added++;\n        }\n\n        distance += segmentDist;\n    }\n\n    return points;\n}\n","'use strict';\n\nvar Point = require('point-geometry');\n\nmodule.exports = {\n    getIcon: getIcon,\n    getGlyphs: getGlyphs\n};\n\nvar minScale = 0.5; // underscale by 1 zoom level\n\nfunction getIcon(anchor, image, boxScale, line, props) {\n\n    var x = image.width / 2 / image.pixelRatio;\n    var y = image.height / 2 / image.pixelRatio;\n\n    var dx = props['icon-offset'][0];\n    var dy = props['icon-offset'][1];\n    var x1 = (dx - x);\n    var x2 = (dx + x);\n    var y1 = (dy - y);\n    var y2 = (dy + y);\n\n    var tl = new Point(x1, y1);\n    var tr = new Point(x2, y1);\n    var br = new Point(x2, y2);\n    var bl = new Point(x1, y2);\n\n    var angle = props['icon-rotate'] * Math.PI / 180;\n    if (anchor.segment !== undefined && props['icon-rotation-alignment'] !== 'viewport') {\n        var next = line[anchor.segment];\n        angle += -Math.atan2(next.x - anchor.x, next.y - anchor.y) + Math.PI / 2;\n    }\n\n    if (angle) {\n        var sin = Math.sin(angle),\n            cos = Math.cos(angle),\n            matrix = [cos, -sin, sin, cos];\n\n        tl = tl.matMult(matrix);\n        tr = tr.matMult(matrix);\n        bl = bl.matMult(matrix);\n        br = br.matMult(matrix);\n\n        x1 = Math.min(tl.x, tr.x, bl.x, br.x);\n        x2 = Math.max(tl.x, tr.x, bl.x, br.x);\n        y1 = Math.min(tl.y, tr.y, bl.y, br.y);\n        y2 = Math.max(tl.y, tr.y, bl.y, br.y);\n    }\n    var box = {\n        x1: x1 * boxScale,\n        x2: x2 * boxScale,\n        y1: y1 * boxScale,\n        y2: y2 * boxScale\n    };\n\n    var iconBox = {\n        box: box,\n        anchor: anchor,\n        minScale: minScale,\n        maxScale: Infinity,\n        padding: props['icon-padding']\n    };\n\n    var icon = {\n        tl: tl,\n        tr: tr,\n        br: br,\n        bl: bl,\n        tex: image,\n        angle: 0,\n        anchor: anchor,\n        minScale: minScale,\n        maxScale: Infinity\n    };\n\n    return {\n        shapes: [icon],\n        boxes: [iconBox],\n        minScale: anchor.scale\n    };\n}\n\nfunction getGlyphs(anchor, origin, shaping, faces, boxScale, horizontal, line, props) {\n\n    var maxAngleDelta = props['text-max-angle'] * Math.PI / 180;\n    var rotate = props['text-rotate'] * Math.PI / 180;\n    var padding = props['text-padding'];\n    var alongLine = props['text-rotation-alignment'] !== 'viewport';\n    var keepUpright = props['text-keep-upright'];\n\n    var glyphs = [],\n        boxes = [];\n\n    var buffer = 3;\n\n    for (var k = 0; k < shaping.length; k++) {\n        var shape = shaping[k];\n        var fontstack = faces[shape.fontstack];\n        var glyph = fontstack.glyphs[shape.glyph];\n        var rect = fontstack.rects[shape.glyph];\n\n        if (!glyph) continue;\n\n        if (!(rect && rect.w > 0 && rect.h > 0)) continue;\n\n        var x = (origin.x + shape.x + glyph.left - buffer + rect.w / 2) * boxScale;\n\n        var glyphInstances;\n        if (anchor.segment !== undefined && alongLine) {\n            glyphInstances = [];\n            getSegmentGlyphs(glyphInstances, anchor, x, line, anchor.segment, 1, maxAngleDelta);\n            if (keepUpright) getSegmentGlyphs(glyphInstances, anchor, x, line, anchor.segment, -1, maxAngleDelta);\n\n        } else {\n            glyphInstances = [{\n                anchor: anchor,\n                offset: 0,\n                angle: 0,\n                maxScale: Infinity,\n                minScale: minScale\n            }];\n        }\n\n        var x1 = origin.x + shape.x + glyph.left - buffer,\n            y1 = origin.y + shape.y - glyph.top - buffer,\n            x2 = x1 + rect.w,\n            y2 = y1 + rect.h,\n\n            otl = new Point(x1, y1),\n            otr = new Point(x2, y1),\n            obl = new Point(x1, y2),\n            obr = new Point(x2, y2);\n\n        var obox = {\n                x1: boxScale * x1,\n                y1: boxScale * y1,\n                x2: boxScale * x2,\n                y2: boxScale * y2\n            };\n\n        for (var i = 0; i < glyphInstances.length; i++) {\n\n            var instance = glyphInstances[i],\n\n                tl = otl,\n                tr = otr,\n                bl = obl,\n                br = obr,\n                box = obox,\n\n                // Clamp to -90/+90 degrees\n                angle = instance.angle + rotate;\n\n            if (angle) {\n                // Compute the transformation matrix.\n                var sin = Math.sin(angle),\n                    cos = Math.cos(angle),\n                    matrix = [cos, -sin, sin, cos];\n\n                tl = tl.matMult(matrix);\n                tr = tr.matMult(matrix);\n                bl = bl.matMult(matrix);\n                br = br.matMult(matrix);\n            }\n\n            // Prevent label from extending past the end of the line\n            var glyphMinScale = Math.max(instance.minScale, anchor.scale);\n\n            // Remember the glyph for later insertion.\n            glyphs.push({\n                tl: tl,\n                tr: tr,\n                bl: bl,\n                br: br,\n                tex: rect,\n                angle: (anchor.angle + rotate + instance.offset + 2 * Math.PI) % (2 * Math.PI),\n                anchor: instance.anchor,\n                minScale: glyphMinScale,\n                maxScale: instance.maxScale\n            });\n\n            if (!instance.offset) { // not a flipped glyph\n                if (angle) {\n                    // Calculate the rotated glyph's bounding box offsets from the anchor point.\n                    box = {\n                        x1: boxScale * Math.min(tl.x, tr.x, bl.x, br.x),\n                        y1: boxScale * Math.min(tl.y, tr.y, bl.y, br.y),\n                        x2: boxScale * Math.max(tl.x, tr.x, bl.x, br.x),\n                        y2: boxScale * Math.max(tl.y, tr.y, bl.y, br.y)\n                    };\n                }\n                boxes.push({\n                    box: box,\n                    anchor: instance.anchor,\n                    minScale: glyphMinScale,\n                    maxScale: instance.maxScale,\n                    padding: padding\n                });\n            }\n        }\n    }\n\n    // TODO avoid creating the boxes in the first place?\n    if (horizontal) boxes = [getMergedBoxes(boxes, anchor)];\n\n    var minPlacementScale = anchor.scale;\n    var minGlyphScale = Infinity;\n    for (var m = 0; m < boxes.length; m++) {\n        minGlyphScale = Math.min(minGlyphScale, boxes[m].minScale);\n    }\n    minGlyphScale = Math.max(minPlacementScale, minScale);\n\n    return {\n        boxes: boxes,\n        shapes: glyphs,\n        minScale: minGlyphScale\n    };\n}\n\nfunction getSegmentGlyphs(glyphs, anchor, offset, line, segment, direction, maxAngleDelta) {\n    var upsideDown = direction < 0;\n\n    if (offset < 0)  direction *= -1;\n\n    if (direction > 0) segment++;\n\n    var newAnchor = anchor;\n    var end = line[segment];\n    var prevscale = Infinity;\n    var prevAngle;\n\n    offset = Math.abs(offset);\n\n    var placementScale = anchor.scale;\n\n    segment_loop:\n    while (true) {\n        var dist = newAnchor.dist(end);\n        var scale = offset/dist;\n        var angle = -Math.atan2(end.x - newAnchor.x, end.y - newAnchor.y) + direction * Math.PI / 2;\n        if (upsideDown) angle += Math.PI;\n\n        // Don't place around sharp corners\n        var angleDiff = (angle - prevAngle) % (2 * Math.PI);\n        if (prevAngle && Math.abs(angleDiff) > maxAngleDelta) {\n            anchor.scale = prevscale;\n            break;\n        }\n\n        glyphs.push({\n            anchor: newAnchor,\n            offset: upsideDown ? Math.PI : 0,\n            minScale: scale,\n            maxScale: prevscale,\n            angle: (angle + 2 * Math.PI) % (2 * Math.PI)\n        });\n\n        if (scale <= placementScale) break;\n\n        newAnchor = end;\n\n        // skip duplicate nodes\n        while (newAnchor.equals(end)) {\n            segment += direction;\n            end = line[segment];\n\n            if (!end) {\n                anchor.scale = scale;\n                break segment_loop;\n            }\n        }\n\n        var unit = end.sub(newAnchor)._unit();\n        newAnchor = newAnchor.sub(unit._mult(dist));\n\n        prevscale = scale;\n        prevAngle = angle;\n    }\n}\n\nfunction getMergedBoxes(glyphs, anchor) {\n      // Collision checks between rotating and fixed labels are relatively expensive,\n      // so we use one box per label, not per glyph for horizontal labels.\n\n    var mergedglyphs = {\n        box: { x1: Infinity, y1: Infinity, x2: -Infinity, y2: -Infinity },\n        anchor: anchor,\n        minScale: 0,\n        padding: -Infinity\n    };\n\n    var box = mergedglyphs.box;\n\n    for (var m = 0; m < glyphs.length; m++) {\n        var gbox = glyphs[m].box;\n        box.x1 = Math.min(box.x1, gbox.x1);\n        box.y1 = Math.min(box.y1, gbox.y1);\n        box.x2 = Math.max(box.x2, gbox.x2);\n        box.y2 = Math.max(box.y2, gbox.y2);\n        mergedglyphs.minScale = Math.max(mergedglyphs.minScale, glyphs[m].minScale);\n        mergedglyphs.padding = Math.max(mergedglyphs.padding, glyphs[m].padding);\n    }\n    // for all horizontal labels, calculate bbox covering all rotated positions\n    var x12 = box.x1 * box.x1,\n        y12 = box.y1 * box.y1,\n        x22 = box.x2 * box.x2,\n        y22 = box.y2 * box.y2,\n        diag = Math.sqrt(Math.max(x12 + y12, x12 + y22, x22 + y12, x22 + y22));\n\n    mergedglyphs.hBox = {\n        x1: -diag,\n        y1: -diag,\n        x2: diag,\n        y2: diag\n    };\n\n    return mergedglyphs;\n}\n","'use strict';\n\nvar resolveTokens = require('../util/token.js');\n\nmodule.exports = resolveText;\n\n// For an array of features determine what glyph ranges need to be loaded\n// and apply any text preprocessing. The remaining users of text should\n// use the `textFeatures` key returned by this function rather than accessing\n// feature text directly.\nfunction resolveText(features, info, glyphs) {\n    var textFeatures = [];\n    var codepoints = [];\n\n    for (var i = 0, fl = features.length; i < fl; i++) {\n        var text = resolveTokens(features[i].properties, info['text-field']);\n        var hastext = false;\n        if (!text) continue;\n        text = text.toString();\n\n        var transform = info['text-transform'];\n        if (transform === 'uppercase') {\n            text = text.toLocaleUpperCase();\n        } else if (transform === 'lowercase') {\n            text = text.toLocaleLowerCase();\n        }\n\n        for (var j = 0, jl = text.length; j < jl; j++) {\n            if (text.charCodeAt(j) <= 65533) {\n                codepoints.push(text.charCodeAt(j));\n                hastext = true;\n            }\n        }\n        // Track indexes of features with text.\n        if (hastext) {\n            textFeatures[i] = text;\n        }\n    }\n\n    // get a list of unique codepoints we are missing\n    codepoints = uniq(codepoints, glyphs);\n\n    return {\n        textFeatures: textFeatures,\n        codepoints: codepoints\n    };\n}\n\nfunction uniq(ids, alreadyHave) {\n    var u = [];\n    var last;\n    ids.sort(sortNumbers);\n    for (var i = 0; i < ids.length; i++) {\n        if (ids[i] !== last) {\n            last = ids[i];\n            if (!alreadyHave[last]) u.push(ids[i]);\n        }\n    }\n    return u;\n}\n\nfunction sortNumbers(a, b) {\n    return a - b;\n}\n\n","'use strict';\n\nvar util = require('../util/util.js'),\n    Point = require('point-geometry');\n\nmodule.exports = {\n    rotationRange: rotationRange,\n    mergeCollisions: mergeCollisions,\n\n    rotatingFixedCollisions: rotatingFixedCollisions,\n    rotatingRotatingCollisions: rotatingRotatingCollisions,\n\n    cornerBoxCollisions: cornerBoxCollisions,\n    circleEdgeCollisions: circleEdgeCollisions,\n\n    getCorners: getCorners,\n};\n\n/*\n * Calculate the range a box conflicts with a second box\n */\nfunction rotationRange(inserting, blocker, scale) {\n\n    var collisions, box;\n\n    var a = inserting;\n    var b = blocker;\n\n    // Instead of scaling the boxes, we move the anchors\n    var relativeAnchor = new Point(\n        (b.anchor.x - a.anchor.x) * scale,\n        (b.anchor.y - a.anchor.y) * scale);\n\n    // Generate a list of collision interval\n    if (a.hBox && b.hBox) {\n        collisions = rotatingRotatingCollisions(a.box, b.box, relativeAnchor);\n\n    } else if (a.hBox) {\n        box = {\n            x1: b.box.x1 + relativeAnchor.x,\n            y1: b.box.y1 + relativeAnchor.y,\n            x2: b.box.x2 + relativeAnchor.x,\n            y2: b.box.y2 + relativeAnchor.y\n        };\n        collisions = rotatingFixedCollisions(a.box, box);\n\n\n    } else if (b.hBox) {\n        box = {\n            x1: a.box.x1 - relativeAnchor.x,\n            y1: a.box.y1 - relativeAnchor.y,\n            x2: a.box.x2 - relativeAnchor.x,\n            y2: a.box.y2 - relativeAnchor.y\n        };\n        collisions = rotatingFixedCollisions(b.box, box);\n\n    } else {\n        collisions = [];\n    }\n\n    // Find and return the continous are around 0 where there are no collisions\n    return mergeCollisions(collisions, blocker.placementRange);\n}\n\n/*\n * Combine an array of collision ranges to form a continuous\n * range that includes 0. Collisions within the ignoreRange are ignored\n */\nfunction mergeCollisions(collisions, ignoreRange) {\n\n    // find continuous interval including 0 that doesn't have any collisions\n    var min = 2 * Math.PI;\n    var max = 0;\n\n    for (var i = 0; i < collisions.length; i++) {\n        var collision = collisions[i];\n\n        var entryOutside = ignoreRange[0] <= collision[0] && collision[0] <= ignoreRange[1];\n        var exitOutside = ignoreRange[0] <= collision[1] && collision[1] <= ignoreRange[1];\n\n        if (entryOutside && exitOutside) {\n            // no collision, since blocker is out of range\n        } else if (entryOutside) {\n            min = Math.min(min, ignoreRange[1]);\n            max = Math.max(max, collision[1]);\n        } else if (exitOutside) {\n            min = Math.min(min, collision[0]);\n            max = Math.max(max, ignoreRange[0]);\n        } else {\n            min = Math.min(min, collision[0]);\n            max = Math.max(max, collision[1]);\n        }\n    }\n\n    return [min, max];\n}\n\n/*\n *  Calculate collision ranges for two rotating boxes.\n */\n\nvar horizontal = new Point(1, 0);\n\nfunction rotatingRotatingCollisions(a, b, anchorToAnchor) {\n    var d = anchorToAnchor.mag();\n\n    var angleBetweenAnchors = anchorToAnchor.angleWith(horizontal);\n\n    var c = [],\n        collisions = [],\n        k;\n\n    // Calculate angles at which collisions may occur\n    // top/bottom\n    c[0] = Math.asin((a.y2 - b.y1) / d);\n    c[1] = Math.asin((a.y2 - b.y1) / d) + Math.PI;\n    c[2] = 2 * Math.PI - Math.asin((-a.y1 + b.y2) / d);\n    c[3] = Math.PI - Math.asin((-a.y1 + b.y2) / d);\n\n    // left/right\n    c[4] = 2 * Math.PI - Math.acos((a.x2 - b.x1) / d);\n    c[5] = Math.acos((a.x2 - b.x1) / d);\n    c[6] = Math.PI - Math.acos((-a.x1 + b.x2) / d);\n    c[7] = Math.PI + Math.acos((-a.x1 + b.x2) / d);\n\n    var rl = a.x2 - b.x1;\n    var lr = -a.x1 + b.x2;\n    var tb = a.y2 - b.y1;\n    var bt = -a.y1 + b.y2;\n\n    // Calculate the distance squared of the diagonal which will be used\n    // to check if the boxes are close enough for collisions to occur at each angle\n    // todo, triple check these\n    var e = [];\n    // top/bottom\n    e[0] = rl * rl + tb * tb;\n    e[1] = lr * lr + tb * tb;\n    e[2] = rl * rl + bt * bt;\n    e[3] = lr * lr + bt * bt;\n    // left/right\n    e[4] = rl * rl + tb * tb;\n    e[5] = rl * rl + bt * bt;\n    e[6] = lr * lr + bt * bt;\n    e[7] = lr * lr + tb * tb;\n\n\n    c = c.filter(function(x, i) {\n        // Check if they are close enough to collide\n        return !isNaN(x) && d * d <= e[i];\n    }).map(function(x) {\n        // So far, angles have been calulated as relative to the vector between anchors.\n        // Convert the angles to angles from north.\n        return (x + angleBetweenAnchors + 2 * Math.PI) % (2 * Math.PI);\n    });\n\n    // Group the collision angles by two\n    // each group represents a range where the two boxes collide\n    c.sort();\n    for (k = 0; k < c.length; k+=2) {\n        collisions.push([c[k], c[k+1]]);\n    }\n\n    return collisions;\n\n}\n\n/*\n *  Calculate collision ranges for a rotating box and a fixed box;\n */\nfunction rotatingFixedCollisions(rotating, fixed) {\n\n    var cornersR = getCorners(rotating);\n    var cornersF = getCorners(fixed);\n\n    // A collision occurs when, and only at least one corner from one of the boxes\n    // is within the other box. Calculate these ranges for each corner.\n\n    var collisions = [];\n\n    for (var i = 0; i < 4; i++ ) {\n        cornerBoxCollisions(collisions, cornersR[i], cornersF);\n        cornerBoxCollisions(collisions, cornersF[i], cornersR, true);\n    }\n\n    return collisions;\n}\n\n\n/*\n *  Calculate the ranges for which the corner,\n *  rotatated around the anchor, is within the box;\n */\nfunction cornerBoxCollisions(collisions, corner, boxCorners, flip) {\n    var radius = corner.mag(),\n        angles = [];\n\n    // Calculate the points at which the corners intersect with the edges\n    for (var i = 0, j = 3; i < 4; j = i++) {\n        circleEdgeCollisions(angles, corner, radius, boxCorners[j], boxCorners[i]);\n    }\n\n    if (angles.length % 2 !== 0) {\n        // TODO fix\n        // This could get hit when a point intersects very close to a corner\n        // and floating point issues cause only one of the entry or exit to be counted\n        throw('expecting an even number of intersections');\n    }\n\n    angles.sort();\n\n    // Group by pairs, where each represents a range where a collision occurs\n    for (var k = 0; k < angles.length; k+=2) {\n        collisions[k/2] = flip ?\n            [2 * Math.PI - angles[k+1], 2 * Math.PI - angles[k]] : // reflect an angle around 0 degrees\n            [angles[k], angles[k+1]];\n    }\n\n    return collisions;\n}\n\n/*\n * Return the intersection points of a circle and a line segment;\n */\nfunction circleEdgeCollisions(angles, corner, radius, p1, p2) {\n\n    var edgeX = p2.x - p1.x;\n    var edgeY = p2.y - p1.y;\n\n    var a = edgeX * edgeX + edgeY * edgeY;\n    var b = (edgeX * p1.x + edgeY * p1.y) * 2;\n    var c = p1.x * p1.x + p1.y * p1.y - radius * radius;\n\n    var discriminant = b*b - 4*a*c;\n\n    // a collision exists only if line intersects circle at two points\n    if (discriminant > 0) {\n        var x1 = (-b - Math.sqrt(discriminant)) / (2 * a);\n        var x2 = (-b + Math.sqrt(discriminant)) / (2 * a);\n\n        // only add points if within line segment\n        // hack to handle floating point representations of 0 and 1\n        if (0 < x1 && x1 < 1) {\n            angles.push(getAngle(p1, p2, x1, corner));\n        }\n\n        if (0 < x2 && x2 < 1) {\n            angles.push(getAngle(p1, p2, x2, corner));\n        }\n    }\n\n    return angles;\n}\n\nfunction getAngle(p1, p2, d, corner) {\n    return (-corner.angleWithSep(\n        util.interp(p1.x, p2.x, d),\n        util.interp(p1.y, p2.y, d)) + 2 * Math.PI) % (2 * Math.PI);\n}\n\nfunction getCorners(a) {\n    return [\n        new Point(a.x1, a.y1),\n        new Point(a.x1, a.y2),\n        new Point(a.x2, a.y2),\n        new Point(a.x2, a.y1)\n    ];\n}\n","'use strict';\n\nmodule.exports = {\n    shape: shape\n};\n\nfunction shape(text, name, stacks, maxWidth, lineHeight, horizontalAlign, verticalAlign, justify, spacing, translate) {\n    var glyphs = stacks[name].glyphs;\n    var glyph;\n\n    var shaping = [];\n\n    var x = translate[0];\n    var y = translate[1];\n    var id;\n\n    for (var i = 0; i < text.length; i++) {\n        id = text.charCodeAt(i);\n        glyph = glyphs[id];\n\n        if (id === 0 || !glyph) continue;\n\n        shaping.push({\n            fontstack: name,\n            glyph: id,\n            x: x,\n            y: y\n        });\n\n        x += glyph.advance + spacing;\n    }\n\n    if (!shaping.length) return false;\n\n    shaping = linewrap(shaping, glyphs, lineHeight, maxWidth, horizontalAlign, verticalAlign, justify);\n\n    return shaping;\n}\n\nvar breakable = { 32: true }; // Currently only breaks at regular spaces\n\nfunction linewrap(shaping, glyphs, lineHeight, maxWidth, horizontalAlign, verticalAlign, justify) {\n    var lastSafeBreak = null;\n\n    var lengthBeforeCurrentLine = 0;\n    var lineStartIndex = 0;\n    var line = 0;\n\n    var maxLineLength = 0;\n\n    if (maxWidth) {\n        for (var i = 0; i < shaping.length; i++) {\n            var shape = shaping[i];\n\n            shape.x -= lengthBeforeCurrentLine;\n            shape.y += lineHeight * line;\n\n            if (shape.x > maxWidth && lastSafeBreak !== null) {\n\n                var lineLength = shaping[lastSafeBreak + 1].x;\n                maxLineLength = Math.max(lineLength, maxLineLength);\n\n                for (var k = lastSafeBreak + 1; k <= i; k++) {\n                    shaping[k].y += lineHeight;\n                    shaping[k].x -= lineLength;\n                }\n\n                if (justify) {\n                    justifyLine(shaping, glyphs, lineStartIndex, lastSafeBreak - 1, justify);\n                }\n\n                lineStartIndex = lastSafeBreak + 1;\n                lastSafeBreak = null;\n                lengthBeforeCurrentLine += lineLength;\n                line++;\n            }\n\n            if (breakable[shape.glyph]) {\n                lastSafeBreak = i;\n            }\n        }\n    }\n\n    maxLineLength = maxLineLength || shaping[shaping.length - 1].x;\n\n    justifyLine(shaping, glyphs, lineStartIndex, shaping.length - 1, justify);\n    align(shaping, justify, horizontalAlign, verticalAlign, maxLineLength, lineHeight, line);\n    return shaping;\n}\n\nfunction justifyLine(shaping, glyphs, start, end, justify) {\n    var lastAdvance = glyphs[shaping[end].glyph].advance;\n    var lineIndent = (shaping[end].x + lastAdvance) * justify;\n\n    for (var j = start; j <= end; j++) {\n        shaping[j].x -= lineIndent;\n    }\n\n}\n\nfunction align(shaping, justify, horizontalAlign, verticalAlign, maxLineLength, lineHeight, line) {\n    var shiftX = (justify - horizontalAlign) * maxLineLength;\n    var shiftY = (-verticalAlign * (line + 1) + 0.5) * lineHeight;\n\n    for (var j = 0; j < shaping.length; j++) {\n        shaping[j].x += shiftX;\n        shaping[j].y += shiftY;\n    }\n}\n","'use strict';\n\nvar Control = require('./control.js'),\n    DOM = require('../../util/dom.js'),\n    util = require('../../util/util.js');\n\nmodule.exports = Attribution;\n\nfunction Attribution() {}\n\nAttribution.prototype = util.inherit(Control, {\n    onAdd: function(map) {\n        var className = 'mapboxgl-ctrl-attrib',\n            container = this._container = DOM.create('div', className, map.container);\n\n        this._update();\n        map.on('source.add', this._update.bind(this));\n        map.on('moveend', this._updateEditLink.bind(this));\n\n        return container;\n    },\n\n    _update: function() {\n        var attributions = [];\n        for (var id in this._map.sources) {\n            var source = this._map.sources[id];\n            if (source.tileJSON && source.tileJSON.attribution) {\n                attributions.push(source.tileJSON.attribution);\n            }\n        }\n        this._container.innerHTML = attributions.join(' | ');\n        this._editLink = this._container.getElementsByClassName('mapbox-improve-map')[0];\n        this._updateEditLink();\n    },\n\n    _updateEditLink: function() {\n        if (this._editLink) {\n            var center = this._map.getCenter();\n            this._editLink.href = 'https://www.mapbox.com/map-feedback/#/' +\n                    center.lng + '/' + center.lat + '/' + Math.round(this._map.getZoom() + 1);\n        }\n    }\n});\n","'use strict';\n\nmodule.exports = Control;\n\nfunction Control() {}\n\nControl.prototype = {\n\taddTo: function(map) {\n\t\tthis._map = map;\n\t\tthis._container = this.onAdd(map);\n\t\treturn this;\n\t},\n\n\tremove: function () {\n\t\tthis._container.parentNode.removeChild(this._container);\n\t\tif (this.onRemove) this.onRemove(this._map);\n\t\tthis._map = null;\n\t\treturn this;\n\t}\n};\n","'use strict';\n\nvar Control = require('./control.js'),\n    DOM = require('../../util/dom.js'),\n    util = require('../../util/util.js');\n\nmodule.exports = Navigation;\n\nfunction Navigation() {}\n\nNavigation.prototype = util.inherit(Control, {\n    onAdd: function(map) {\n        var className = 'mapboxgl-ctrl-nav';\n\n        var container = this._container = DOM.create('div', className, map.container);\n\n        this._zoomInButton = this._createButton(className + '-zoom-in', map.zoomIn.bind(map));\n        this._zoomOutButton = this._createButton(className + '-zoom-out', map.zoomOut.bind(map));\n        this._compass = this._createButton(className + '-compass', map.resetNorth.bind(map));\n\n        var compassCanvas = this._compassCanvas = DOM.create('canvas', className + '-compass-canvas', this._compass);\n        compassCanvas.style.cssText = 'width:26px; height:26px;';\n        compassCanvas.width = 26 * 2;\n        compassCanvas.height = 26 * 2;\n\n        this._compass.addEventListener('mousedown', this._onCompassDown.bind(this));\n        this._onCompassMove = this._onCompassMove.bind(this);\n        this._onCompassUp = this._onCompassUp.bind(this);\n\n        this._compassCtx = compassCanvas.getContext('2d');\n\n        map.on('rotate', this._drawNorth.bind(this));\n        this._drawNorth();\n\n        return container;\n    },\n\n    _onCompassDown: function(e) {\n        DOM.disableDrag();\n\n        document.addEventListener('mousemove', this._onCompassMove);\n        document.addEventListener('mouseup', this._onCompassUp);\n        this._prevX = e.screenX;\n\n        e.stopPropagation();\n    },\n\n    _onCompassMove: function(e) {\n        var x = e.screenX,\n            d = x < 2 ? -5 : // left edge of the screen, continue rotating\n                x > window.screen.width - 2 ? 5 : // right edge\n                (x - this._prevX) / 4;\n\n        this._map.setBearing(this._map.getBearing() - d);\n        this._prevX = e.screenX;\n\n        e.preventDefault();\n    },\n\n    _onCompassUp: function() {\n        document.removeEventListener('mousemove', this._onCompassMove);\n        document.removeEventListener('mouseup', this._onCompassUp);\n        DOM.enableDrag();\n    },\n\n    _createButton: function(className, fn) {\n        var a = DOM.create('a', className, this._container);\n        a.href = '#';\n        a.addEventListener('click', function (e) {\n            fn();\n            e.preventDefault();\n            e.stopPropagation();\n        });\n        return a;\n    },\n\n    _drawNorth: function() {\n        var rad = 20,\n            width = 8,\n            center = 26,\n            angle = this._map.transform.angle + (Math.PI / 2),\n            ctx = this._compassCtx;\n\n        this._compassCanvas.width = this._compassCanvas.width;\n\n        ctx.translate(center, center);\n        ctx.rotate(angle);\n\n        ctx.beginPath();\n        ctx.fillStyle = '#000';\n        ctx.lineTo(0, -width);\n        ctx.lineTo(-rad, 0);\n        ctx.lineTo(0, width);\n        ctx.fill();\n\n        ctx.beginPath();\n        ctx.fillStyle = '#bbb';\n        ctx.moveTo(0, 0);\n        ctx.lineTo(0, width);\n        ctx.lineTo(rad, 0);\n        ctx.lineTo(0, -width);\n        ctx.fill();\n\n        ctx.beginPath();\n        ctx.strokeStyle = '#fff';\n        ctx.lineWidth = 4;\n        ctx.moveTo(0, -width);\n        ctx.lineTo(0, width);\n        ctx.stroke();\n    }\n});\n","'use strict';\n\nvar util = require('../util/util.js'),\n    browser = require('../util/browser.js'),\n    LatLng = require('../geo/latlng.js'),\n    LatLngBounds = require('../geo/latlngbounds.js'),\n    Point = require('point-geometry');\n\nutil.extend(exports, {\n    isEasing: function () {\n        return !!this._stopFn;\n    },\n\n    stop: function () {\n        if (this._stopFn) {\n            this._stopFn();\n            delete this._stopFn;\n        }\n        return this;\n    },\n\n    panBy: function(offset, options) {\n        this.panTo(this.transform.center, util.extend({offset: Point.convert(offset).mult(-1)}, options));\n        return this;\n    },\n\n    panTo: function(latlng, options) {\n        this.stop();\n\n        latlng = LatLng.convert(latlng);\n\n        options = util.extend({\n            duration: 500,\n            easing: util.ease,\n            offset: [0, 0]\n        }, options);\n\n        var tr = this.transform,\n            offset = Point.convert(options.offset).rotate(-tr.angle),\n            from = tr.point,\n            to = tr.project(latlng).sub(offset);\n\n        if (!options.noMoveStart) {\n            this.fire('movestart');\n        }\n\n        this._stopFn = browser.timed(function(t) {\n            tr.center = tr.unproject(from.add(to.sub(from).mult(options.easing(t))));\n            this._move();\n            if (t === 1) this.fire('moveend');\n\n        }, options.animate === false ? 0 : options.duration, this);\n\n        return this;\n    },\n\n    // Zooms to a certain zoom level with easing.\n    zoomTo: function(zoom, options) {\n        this.stop();\n\n        options = util.extend({\n            duration: 500\n        }, options);\n\n        var tr = this.transform,\n            around = tr.center,\n            easing = this._updateEasing(options.duration, zoom, options.easing),\n            startZoom = tr.zoom;\n\n        if (options.around) {\n            around = LatLng.convert(options.around);\n        } else if (options.offset) {\n            around = tr.pointLocation(tr.centerPoint.add(Point.convert(options.offset)));\n        }\n\n        if (options.animate === false) options.duration = 0;\n\n        if (!this.zooming) {\n            this.zooming = true;\n            this.fire('movestart');\n        }\n\n        this._stopFn = browser.timed(function(t) {\n            tr.setZoomAround(util.interp(startZoom, zoom, easing(t)), around);\n\n            this.style.animationLoop.set(300); // text fading\n            this._move(true);\n\n            if (t === 1) {\n                this.ease = null;\n                if (options.duration >= 200) {\n                    this.fire('moveend');\n                    this.zooming = false;\n                }\n            }\n\n        }, options.duration, this);\n\n        if (options.duration < 200) {\n            clearTimeout(this._onZoomEnd);\n            this._onZoomEnd = setTimeout(function() {\n                this.zooming = false;\n                this._rerender();\n                this.fire('moveend');\n            }.bind(this), 200);\n        }\n\n        return this;\n    },\n\n    zoomIn: function(options) {\n        this.zoomTo(this.getZoom() + 1, options);\n    },\n\n    zoomOut: function(options) {\n        this.zoomTo(this.getZoom() - 1, options);\n    },\n\n    rotateTo: function(bearing, options) {\n        this.stop();\n\n        options = util.extend({\n            duration: 500,\n            easing: util.ease\n        }, options);\n\n        var tr = this.transform,\n            start = this.getBearing(),\n            around = tr.center;\n\n        if (options.around) {\n            around = LatLng.convert(options.around);\n        } else if (options.offset) {\n            around = tr.pointLocation(tr.centerPoint.add(Point.convert(options.offset)));\n        }\n\n        this.rotating = true;\n        this.fire('movestart');\n\n        this._stopFn = browser.timed(function(t) {\n            if (t === 1) { this.rotating = false; }\n            tr.setBearingAround(util.interp(start, bearing, options.easing(t)), around);\n            this._move(false, true).fire('moveend');\n        }, options.animate === false ? 0 : options.duration, this);\n\n        return this;\n    },\n\n    resetNorth: function(options) {\n        return this.rotateTo(0, util.extend({duration: 1000}, options));\n    },\n\n    fitBounds: function(bounds, options) {\n\n        options = util.extend({\n            padding: 0,\n            offset: [0, 0],\n            maxZoom: Infinity\n        }, options);\n\n        bounds = LatLngBounds.convert(bounds);\n\n        var offset = Point.convert(options.offset),\n            tr = this.transform,\n            nw = tr.project(bounds.getNorthWest()),\n            se = tr.project(bounds.getSouthEast()),\n            size = se.sub(nw),\n            center = tr.unproject(nw.add(se).div(2)),\n\n            scaleX = (tr.width - options.padding * 2 - Math.abs(offset.x) * 2) / size.x,\n            scaleY = (tr.height - options.padding * 2 - Math.abs(offset.y) * 2) / size.y,\n\n            zoom = Math.min(tr.scaleZoom(tr.scale * Math.min(scaleX, scaleY)), options.maxZoom);\n\n        return options.linear ?\n            this.easeTo(center, zoom, 0, options) :\n            this.flyTo(center, zoom, 0, options);\n    },\n\n    easeTo: function(latlng, zoom, bearing, options) {\n\n        options = util.extend({\n            offset: [0, 0],\n            duration: 500,\n            easing: util.ease\n        }, options);\n\n        latlng = LatLng.convert(latlng);\n\n        var offset = Point.convert(options.offset),\n            tr = this.transform,\n            startZoom = this.getZoom(),\n            startBearing = this.getBearing();\n\n        zoom = zoom === undefined ? startZoom : zoom;\n        bearing = bearing === undefined ? startBearing : bearing;\n\n        var scale = tr.zoomScale(zoom - startZoom),\n            from = tr.point,\n            to = tr.project(latlng).sub(offset.div(scale)),\n            around = tr.pointLocation(tr.centerPoint.add(to.sub(from).div(1 - 1 / scale)));\n\n        if (zoom !== startZoom) this.zooming = true;\n        if (startBearing !== bearing) this.rotating = true;\n\n        this.fire('movestart');\n\n        this._stopFn = browser.timed(function (t) {\n            var k = options.easing(t);\n\n            if (zoom !== startZoom) {\n                tr.setZoomAround(startZoom + k * (zoom - startZoom), around);\n            }\n            if (bearing !== startBearing) {\n                tr.bearing = util.interp(startBearing, bearing, k);\n            }\n\n            this.style.animationLoop.set(300); // text fading\n            this._move(zoom !== startZoom, bearing !== startBearing);\n\n            if (t === 1) {\n                this.zooming = false;\n                this.rotating = false;\n                this.fire('moveend');\n            }\n\n        }, options.animate === false ? 0 : options.duration, this);\n\n        return this;\n    },\n\n    flyTo: function(latlng, zoom, bearing, options) {\n\n        options = util.extend({\n            offset: [0, 0],\n            speed: 1.2,\n            curve: 1.42,\n            easing: util.ease\n        }, options);\n\n        latlng = LatLng.convert(latlng);\n\n        var offset = Point.convert(options.offset),\n            tr = this.transform,\n            startZoom = this.getZoom(),\n            startBearing = this.getBearing();\n\n        zoom = zoom === undefined ? startZoom : zoom;\n        bearing = bearing === undefined ? startBearing : bearing;\n\n        var scale = tr.zoomScale(zoom - startZoom),\n            from = tr.point,\n            to = tr.project(latlng).sub(offset.div(scale));\n\n        if (options.animate === false) {\n            return this.setView(latlng, zoom, bearing);\n        }\n\n        var startWorldSize = tr.worldSize,\n            rho = options.curve,\n            V = options.speed,\n\n            w0 = Math.max(tr.width, tr.height),\n            w1 = w0 / scale,\n            u1 = to.sub(from).mag(),\n            rho2 = rho * rho;\n\n        function r(i) {\n            var b = (w1 * w1 - w0 * w0 + (i ? -1 : 1) * rho2 * rho2 * u1 * u1) / (2 * (i ? w1 : w0) * rho2 * u1);\n            return Math.log(Math.sqrt(b * b + 1) - b);\n        }\n\n        function sinh(n) { return (Math.exp(n) - Math.exp(-n)) / 2; }\n        function cosh(n) { return (Math.exp(n) + Math.exp(-n)) / 2; }\n        function tanh(n) { return sinh(n) / cosh(n); }\n\n        var r0 = r(0),\n            w = function (s) { return (cosh(r0) / cosh(r0 + rho * s)); },\n            u = function (s) { return w0 * ((cosh(r0) * tanh(r0 + rho * s) - sinh(r0)) / rho2) / u1; },\n            S = (r(1) - r0) / rho;\n\n        if (Math.abs(u1) < 0.000001) {\n            if (Math.abs(w0 - w1) < 0.000001) return this;\n\n            var k = w1 < w0 ? -1 : 1;\n            S = Math.abs(Math.log(w1 / w0)) / rho;\n\n            u = function() { return 0; };\n            w = function(s) { return Math.exp(k * rho * s); };\n        }\n\n        var duration = 1000 * S / V;\n\n        this.zooming = true;\n        if (startBearing != bearing) this.rotating = true;\n\n        this.fire('movestart');\n\n        this._stopFn = browser.timed(function (t) {\n            var k = options.easing(t),\n                s = k * S,\n                us = u(s);\n\n            tr.zoom = startZoom + tr.scaleZoom(1 / w(s));\n            tr.center = tr.unproject(from.add(to.sub(from).mult(us)), startWorldSize);\n\n            if (bearing !== startBearing) {\n                tr.bearing = util.interp(startBearing, bearing, k);\n            }\n\n            this.style.animationLoop.set(300); // text fading\n\n            this._move(true, bearing !== startBearing);\n\n            if (t === 1) {\n                this.zooming = false;\n                this.rotating = false;\n                this.fire('moveend');\n            }\n        }, duration, this);\n\n        return this;\n    },\n\n    _updateEasing: function(duration, zoom, bezier) {\n        var easing;\n\n        if (this.ease) {\n            var ease = this.ease,\n                t = (Date.now() - ease.start) / ease.duration,\n                speed = ease.easing(t + 0.01) - ease.easing(t),\n\n                // Quick hack to make new bezier that is continuous with last\n                x = 0.27 / Math.sqrt(speed * speed + 0.0001) * 0.01,\n                y = Math.sqrt(0.27 * 0.27 - x * x);\n\n            easing = util.bezier(x, y, 0.25, 1);\n        } else {\n            easing = bezier ? util.bezier.apply(util, bezier) : util.ease;\n        }\n\n        // store information on current easing\n        this.ease = {\n            start: (new Date()).getTime(),\n            to: Math.pow(2, zoom),\n            duration: duration,\n            easing: easing\n        };\n\n        return easing;\n    }\n});\n","'use strict';\n\nvar Interaction = require('./interaction.js');\nvar Point = require('point-geometry');\nvar util = require('../util/util.js');\n\nmodule.exports = Handlers;\n\nfunction Handlers(map) {\n\n    var rotateEnd;\n\n    var inertiaLinearity = 0.2,\n        inertiaEasing = util.bezier(0, 0, inertiaLinearity, 1);\n\n    this.interaction = new Interaction(map.container)\n        .on('click', function(e) {\n            map.fire('click', e);\n        })\n        .on('hover', function(e) {\n            map.fire('hover', e);\n        })\n        .on('down', function () {\n            map.fire('movestart');\n        })\n        .on('resize', function() {\n            map.stop();\n            map.resize();\n            map.update();\n        })\n        .on('pan', function(e) {\n            map.stop();\n            map.transform.panBy(e.offset);\n            map._move();\n        })\n        .on('panend', function(e) {\n            if (!e.inertia) map.fire('moveend');\n            else {\n                // convert velocity to px/s & adjust for increased initial animation speed when easing out\n                var velocity = e.inertia.mult(1000 * inertiaLinearity),\n                    speed = velocity.mag();\n\n                var maxSpeed = 4000; // px/s\n\n                if (speed >= maxSpeed) {\n                    speed = maxSpeed;\n                    velocity._unit()._mult(maxSpeed);\n                }\n\n                var deceleration = 8000, // px/s^2\n                    duration = speed / (deceleration * inertiaLinearity),\n                    offset = velocity.mult(-duration / 2).round();\n\n                map.panBy(offset, {\n                    duration: duration * 1000,\n                    easing: inertiaEasing,\n                    noMoveStart: true\n                });\n            }\n        })\n        .on('zoom', function(e) {\n            // Scale by sigmoid of scroll wheel delta.\n            var scale = 2 / (1 + Math.exp(-Math.abs(e.delta / 100)));\n            if (e.delta < 0 && scale !== 0) scale = 1 / scale;\n\n            var fromScale = map.ease && isFinite(e.delta) ? map.ease.to : map.transform.scale,\n                duration = !isFinite(e.delta) ? 800 : e.source == 'trackpad' ? 0 : 300;\n\n            map.zoomTo(map.transform.scaleZoom(fromScale * scale), {\n                duration: duration,\n                around: map.unproject(e.point)\n            });\n        })\n        .on('rotate', function(e) {\n            var center = map.transform.centerPoint, // Center of rotation\n                startToCenter = e.start.sub(center),\n                startToCenterDist = startToCenter.mag();\n\n            // If the first click was too close to the center, move the center of rotation by 200 pixels\n            // in the direction of the click.\n            if (startToCenterDist < 200) {\n                center = e.start.add(new Point(-200, 0)._rotate(startToCenter.angle()));\n            }\n\n            var bearingDiff = e.prev.sub(center).angleWith(e.current.sub(center)) / Math.PI * 180;\n            map.transform.bearing = map.getBearing() - bearingDiff;\n\n            map._move(false, true);\n\n            window.clearTimeout(rotateEnd);\n            rotateEnd = window.setTimeout(function() {\n                map.rotating = false;\n                map._rerender();\n            }, 200);\n        });\n}\n","'use strict';\n\nmodule.exports = Hash;\n\nvar util = require('../util/util.js');\n\nfunction Hash(map) {\n    this.map = map;\n    window.addEventListener('hashchange', this.onhash.bind(this), false);\n    map.on('move', util.debounce(this.updateHash.bind(this), 100));\n}\n\nHash.prototype = {\n    onhash: function() {\n        var loc = location.hash.replace('#', '').split('/');\n        if (loc.length >= 3) {\n            this.map.setView([+loc[1], +loc[2]], +loc[0], +(loc[3] || 0));\n            return true;\n        }\n        return false;\n    },\n\n    updateHash: function() {\n        var center = this.map.getCenter(),\n            zoom = this.map.getZoom(),\n            bearing = this.map.getBearing(),\n            precision = Math.max(0, Math.ceil(Math.log(zoom) / Math.LN2)),\n\n            hash = '#' + (Math.round(zoom * 100) / 100) +\n                '/' + center.lat.toFixed(precision) +\n                '/' + center.lng.toFixed(precision) +\n                (bearing ? '/' + (Math.round(bearing * 10) / 10) : '');\n\n        window.history.replaceState('', '', hash);\n    }\n};\n","'use strict';\n\nvar Evented = require('../util/evented.js'),\n    browser = require('../util/browser.js'),\n    Point = require('point-geometry');\n\nmodule.exports = Interaction;\n\nfunction Interaction(el) {\n    var interaction = this;\n    if (!el) return;\n\n    var rotating = false,\n        panned = false,\n        firstPos = null,\n        pos = null,\n        inertia = null,\n        now;\n\n    function mousePos(e) {\n        var rect = el.getBoundingClientRect();\n        return new Point(\n            e.clientX - rect.left - el.clientLeft,\n            e.clientY - rect.top - el.clientTop);\n    }\n\n    el.addEventListener('contextmenu', function(ev) {\n        rotating = true;\n        firstPos = pos = mousePos(ev);\n        ev.preventDefault();\n    }, false);\n    el.addEventListener('mousedown', onmousedown, false);\n    document.addEventListener('mouseup', onmouseup, false);\n    document.addEventListener('mousemove', onmousemove, false);\n    el.addEventListener('click', onclick, false);\n    scrollwheel(zoom);\n    el.addEventListener('dblclick', ondoubleclick, false);\n    window.addEventListener('resize', resize, false);\n\n    function zoom(type, delta, point) {\n        interaction.fire('zoom', {\n            source: type,\n            delta: delta,\n            point: point\n        });\n        inertia = null;\n        now = null;\n    }\n\n    function click(point) {\n        interaction.fire('click', {point: point});\n    }\n\n    function hover(point) {\n        interaction.fire('hover', {point: point});\n    }\n\n    function pan(point) {\n        if (pos) {\n            var offset = pos.sub(point);\n            interaction.fire('pan', {offset: offset});\n\n            // add an averaged version of this movement to the inertia vector\n            if (inertia) {\n                var duration = Date.now() - now;\n                // sometimes it's 0 after some erratic paning\n                if (duration) {\n                    var time = duration + now;\n                    inertia.push([time, point]);\n                    while (inertia.length > 2 && time - inertia[0][0] > 100) inertia.shift();\n                }\n\n            } else {\n                inertia = [];\n            }\n            now = Date.now();\n            pos = point;\n        }\n    }\n\n    function resize() {\n        interaction.fire('resize');\n    }\n\n    function rotate(point) {\n        if (pos) {\n            interaction.fire('rotate', {\n                start: firstPos,\n                prev: pos,\n                current: point\n            });\n            pos = point;\n        }\n    }\n\n    function onmousedown(ev) {\n        firstPos = pos = mousePos(ev);\n        interaction.fire('down');\n    }\n\n    function onmouseup() {\n        panned = pos && firstPos && (pos.x != firstPos.x || pos.y != firstPos.y);\n\n        rotating = false;\n        pos = null;\n\n        if (inertia && inertia.length >= 2 && now > Date.now() - 100) {\n            var last = inertia[inertia.length - 1],\n                first = inertia[0],\n                velocity = last[1].sub(first[1]).div(last[0] - first[0]);\n            interaction.fire('panend',  {inertia: velocity});\n\n        } else interaction.fire('panend');\n\n        inertia = null;\n        now = null;\n    }\n\n    function onmousemove(ev) {\n        var point = mousePos(ev);\n\n        if (rotating) { rotate(point); }\n        else if (pos) pan(point);\n        else {\n            var target = ev.toElement;\n            while (target && target != el && target.parentNode) target = target.parentNode;\n            if (target == el) {\n                hover(point);\n            }\n        }\n    }\n\n    function onclick(ev) {\n        if (!panned) click(mousePos(ev));\n    }\n\n    function ondoubleclick(ev) {\n        zoom('wheel', Infinity * (ev.shiftKey ? -1 : 1), mousePos(ev));\n        ev.preventDefault();\n    }\n\n    function scrollwheel(callback) {\n        var firefox = /Firefox/i.test(navigator.userAgent);\n        var safari = /Safari/i.test(navigator.userAgent) && !/Chrom(ium|e)/i.test(navigator.userAgent);\n        var time = window.performance || Date;\n\n        el.addEventListener('wheel', wheel, false);\n        el.addEventListener('mousewheel', mousewheel, false);\n\n        var lastEvent = 0;\n\n        var type = null;\n        var typeTimeout = null;\n        var initialValue = null;\n\n        function scroll(value, ev) {\n            var stamp = time.now();\n            var timeDelta = stamp - lastEvent;\n            lastEvent = stamp;\n\n            var point = mousePos(ev);\n\n            if (value !== 0 && (value % 4.000244140625) === 0) {\n                // This one is definitely a mouse wheel event.\n                type = 'wheel';\n            } else if (value !== 0 && Math.abs(value) < 4) {\n                // This one is definitely a trackpad event because it is so small.\n                type = 'trackpad';\n            } else if (timeDelta > 400) {\n                // This is likely a new scroll action.\n                type = null;\n                initialValue = value;\n                // Start a timeout in case this was a singular event, and dely it\n                // by up to 40ms.\n                typeTimeout = setTimeout(function() {\n                    type = 'wheel';\n                    callback(type, -initialValue, point);\n                }, 40);\n            } else if (type === null) {\n                // This is a repeating event, but we don't know the type of event\n                // just yet. If the delta per time is small, we assume it's a\n                // fast trackpad; otherwise we switch into wheel mode.\n                type = (Math.abs(timeDelta * value) < 200) ? 'trackpad' : 'wheel';\n\n                // Make sure our delayed event isn't fired again, because we\n                // accumulate the previous event (which was less than 40ms ago) into\n                // this event.\n                if (typeTimeout) {\n                    clearTimeout(typeTimeout);\n                    typeTimeout = null;\n                    value += initialValue;\n                }\n            }\n\n            // Only fire the callback if we actually know what type of scrolling\n            // device the user uses.\n            if (type !== null) {\n                callback(type, -value, point);\n            }\n        }\n\n        function wheel(e) {\n            var deltaY = e.deltaY;\n            // Firefox doubles the values on retina screens...\n            if (firefox && e.deltaMode == window.WheelEvent.DOM_DELTA_PIXEL) deltaY /= browser.devicePixelRatio;\n            if (e.deltaMode == window.WheelEvent.DOM_DELTA_LINE) deltaY *= 40;\n            scroll(deltaY, e);\n            e.preventDefault();\n        }\n\n        function mousewheel(e) {\n            var deltaY = -e.wheelDeltaY;\n            if (safari) deltaY = deltaY / 3;\n            scroll(deltaY, e);\n            e.preventDefault();\n        }\n    }\n}\n\nInteraction.prototype = Object.create(Evented);\n","'use strict';\n\nvar Dispatcher = require('../util/dispatcher.js'),\n    Canvas = require('../util/canvas.js'),\n    util = require('../util/util.js'),\n    browser = require('../util/browser.js'),\n    ajax = require('../util/ajax.js'),\n    Evented = require('../util/evented.js'),\n\n    Style = require('../style/style.js'),\n    AnimationLoop = require('../style/animationloop.js'),\n    GLPainter = require('../render/painter.js'),\n\n    Transform = require('../geo/transform.js'),\n    Hash = require('./hash.js'),\n    Handlers = require('./handlers.js'),\n    Source = require('../source/source.js'),\n    Easings = require('./easings.js'),\n    LatLng = require('../geo/latlng.js'),\n    LatLngBounds = require('../geo/latlngbounds.js'),\n    Point = require('point-geometry'),\n    GlyphSource = require('../symbol/glyphsource.js'),\n    Attribution = require('./control/attribution.js');\n\n// allow redefining Map here (jshint thinks it's global)\n// jshint -W079\n\nvar Map = module.exports = function(options) {\n\n    options = this.options = util.inherit(this.options, options);\n\n    this.animationLoop = new AnimationLoop();\n    this.transform = new Transform(options.minZoom, options.maxZoom);\n    this.hash = options.hash && new Hash(this);\n\n    if (options.maxBounds) {\n        var b = LatLngBounds.convert(options.maxBounds);\n        this.transform.latRange = [b.getSouth(), b.getNorth()];\n        this.transform.lngRange = [b.getWest(), b.getEast()];\n    }\n\n    this._onStyleChange = this._onStyleChange.bind(this);\n    this._updateBuckets = this._updateBuckets.bind(this);\n    this.render = this.render.bind(this);\n\n    this._setupContainer();\n    this._setupPainter();\n\n    this.handlers = options.interactive && new Handlers(this);\n    this.dispatcher = new Dispatcher(Math.max(options.numWorkers, 1), this);\n\n     // don't set position from options if set through hash\n    if (!this.hash || !this.hash.onhash()) {\n        this.setView(options.center, options.zoom, options.bearing);\n    }\n\n    this.sources = {};\n    this.stacks = {};\n\n    this.resize();\n\n    if (typeof options.style === 'object') {\n        this.setStyle(options.style);\n\n    } else if (typeof options.style === 'string') {\n        ajax.getJSON(options.style, function (err, data) {\n            if (err) throw err;\n            this.setStyle(data);\n        }.bind(this));\n    }\n\n    if (options.attributionControl) this.addControl(new Attribution());\n};\n\nutil.extend(Map.prototype, Evented);\nutil.extend(Map.prototype, Easings);\nutil.extend(Map.prototype, {\n\n    options: {\n        center: [0, 0],\n        zoom: 0,\n        bearing: 0,\n\n        minZoom: 0,\n        maxZoom: 20,\n        numWorkers: browser.hardwareConcurrency - 1,\n\n        interactive: true,\n        hash: false,\n\n        attributionControl: true\n    },\n\n    addSource: function(id, source) {\n        this.sources[id] = source;\n        source.id = id;\n        if (source.onAdd) {\n            source.onAdd(this);\n        }\n        if (source.enabled) source.fire('source.add', {source: source});\n        return this;\n    },\n\n    removeSource: function(id) {\n        var source = this.sources[id];\n        if (source.onRemove) {\n            source.onRemove(this);\n        }\n        delete this.sources[id];\n        return this.fire('source.remove', {source: source});\n    },\n\n    addControl: function(control) {\n        control.addTo(this);\n        return this;\n    },\n\n    // Set the map's center, zoom, and bearing\n    setView: function(center, zoom, bearing) {\n        this.stop();\n\n        var tr = this.transform,\n            zoomChanged = tr.zoom !== +zoom,\n            bearingChanged = tr.bearing !== +bearing;\n\n        tr.center = LatLng.convert(center);\n        tr.zoom = +zoom;\n        tr.bearing = +bearing;\n\n        return this\n            .fire('movestart')\n            ._move(zoomChanged, bearingChanged)\n            .fire('moveend');\n    },\n\n    setCenter: function(center) {\n        this.setView(center, this.getZoom(), this.getBearing());\n    },\n\n    setZoom: function(zoom) {\n        this.setView(this.getCenter(), zoom, this.getBearing());\n    },\n\n    setBearing: function(bearing) {\n        this.setView(this.getCenter(), this.getZoom(), bearing);\n    },\n\n    getCenter: function() { return this.transform.center; },\n    getZoom: function() { return this.transform.zoom; },\n    getBearing: function() { return this.transform.bearing; },\n\n    // Detect the map's new width and height and resize it.\n    resize: function() {\n        var width = 0, height = 0;\n\n        if (this.container) {\n            width = this.container.offsetWidth || 400;\n            height = this.container.offsetHeight || 300;\n        }\n\n        this.canvas.resize(width, height);\n\n        this.transform.width = width;\n        this.transform.height = height;\n        this.transform._constrain();\n\n        if (this.style && this.style.sprite) {\n            this.style.sprite.resize(this.painter.gl);\n        }\n\n        this.painter.resize(width, height);\n\n        return this\n            .fire('movestart')\n            ._move()\n            .fire('resize')\n            .fire('moveend');\n    },\n\n    getBounds: function() {\n        return new LatLngBounds(\n            this.transform.pointLocation(new Point(0, 0)),\n            this.transform.pointLocation(this.transform.size));\n    },\n\n    project: function(latlng) {\n        return this.transform.locationPoint(LatLng.convert(latlng));\n    },\n    unproject: function(point) {\n        return this.transform.pointLocation(Point.convert(point));\n    },\n\n    featuresAt: function(point, params, callback) {\n        var features = [];\n        var error = null;\n        var map = this;\n\n        point = Point.convert(point);\n\n        util.asyncEach(Object.keys(this.sources), function(id, callback) {\n            var source = map.sources[id];\n            source.featuresAt(point, params, function(err, result) {\n                if (result) features = features.concat(result);\n                if (err) error = err;\n                callback();\n            });\n        }, function() {\n            callback(error, features);\n        });\n        return this;\n    },\n\n    setStyle: function(style) {\n        if (this.style) {\n            this.style.off('change', this._onStyleChange);\n        }\n\n        if (style instanceof Style) {\n            this.style = style;\n        } else {\n            this.style = new Style(style, this.animationLoop);\n        }\n\n        var sources = this.style.stylesheet.sources;\n        for (var id in sources) {\n            this.addSource(id, Source.create(sources[id]));\n        }\n\n        this.glyphSource = new GlyphSource(this.style.stylesheet.glyphs, this.painter.glyphAtlas);\n\n        this.style.on('change', this._onStyleChange);\n\n        this._styleDirty = true;\n        this._tilesDirty = true;\n\n        this._updateBuckets();\n        this._updateGlyphs();\n\n        this.fire('style.change');\n\n        return this;\n    },\n\n    _move: function (zoom, rotate) {\n\n        this.update(zoom).fire('move');\n\n        if (zoom) this.fire('zoom');\n        if (rotate) this.fire('rotate');\n\n        return this;\n    },\n\n    // map setup code\n\n    _setupContainer: function() {\n        var id = this.options.container;\n        var container = this.container = typeof id === 'string' ? document.getElementById(id) : id;\n        if (container) container.classList.add('mapboxgl-map');\n        this.canvas = new Canvas(this, container);\n    },\n\n    _setupPainter: function() {\n        var gl = this.canvas.getWebGLContext();\n\n        if (!gl) {\n            alert('Failed to initialize WebGL');\n            return;\n        }\n\n        this.painter = new GLPainter(gl, this.transform);\n    },\n\n    _contextLost: function(event) {\n        event.preventDefault();\n        if (this._frameId) {\n            browser.cancelFrame(this._frameId);\n        }\n    },\n\n    _contextRestored: function() {\n        this._setupPainter();\n        this.resize();\n        this.update();\n    },\n\n    // Callbacks from web workers\n\n    'debug message': function(data) {\n        console.log.apply(console, data);\n    },\n\n    'alert message': function(data) {\n        alert.apply(window, data);\n    },\n\n    'get sprite json': function(params, callback) {\n        var sprite = this.style.sprite;\n        if (sprite.loaded()) {\n            callback(null, { sprite: sprite.data, retina: sprite.retina });\n        } else {\n            sprite.on('loaded', function() {\n                callback(null, { sprite: sprite.data, retina: sprite.retina });\n            });\n        }\n    },\n\n    'get glyphs': function(params, callback) {\n        this.glyphSource.getRects(params.fontstack, params.codepoints, params.id, callback);\n    },\n\n    // Rendering\n\n    update: function(updateStyle) {\n\n        if (!this.style) return this;\n\n        this._styleDirty = this._styleDirty || updateStyle;\n        this._tilesDirty = true;\n\n        this._rerender();\n\n        return this;\n    },\n\n    // Call when a (re-)render of the map is required, e.g. when the user panned or zoomed,f or new data is available.\n    render: function() {\n        if (this._styleDirty) {\n            this._styleDirty = false;\n            this._updateStyle();\n        }\n\n        if (this._tilesDirty) {\n            for (var id in this.sources) {\n                this.sources[id].update();\n            }\n            this._tilesDirty = false;\n        }\n\n        this._renderGroups(this.style.layerGroups);\n        this.fire('render');\n\n        this._frameId = null;\n\n        if (!this.animationLoop.stopped()) {\n            this._styleDirty = true;\n        }\n\n        if (this._repaint || !this.animationLoop.stopped()) {\n            this._rerender();\n        }\n\n        return this;\n    },\n\n    _renderGroups: function(groups, name) {\n\n        var i, len, group, source, k;\n\n        // Render all dependencies (composited layers) to textures\n        for (i = 0, len = groups.length; i < len; i++) {\n            group = groups[i];\n\n            for (k in group.dependencies) {\n                this._renderGroups(group.dependencies[k], k);\n            }\n        }\n\n        // attach render destination. if no name, main canvas.\n        this.painter.bindRenderTexture(name);\n\n        // Render the groups\n        for (i = 0, len = groups.length; i < len; i++) {\n            group = groups[i];\n            source = this.sources[group.source];\n\n            if (source) {\n                this.painter.clearStencil();\n                source.render(group);\n\n            } else if (group.composited) {\n                this.painter.draw(undefined, this.style, group, {});\n            } else if (group.source === undefined) {\n                this.painter.draw(undefined, this.style, group, { background: true });\n            }\n        }\n    },\n\n    _rerender: function() {\n        if (!this._frameId) {\n            this._frameId = browser.frame(this.render);\n        }\n    },\n\n    _onStyleChange: function () {\n        this.update(true);\n    },\n\n    _updateStyle: function() {\n        if (!this.style) return;\n        this.style.recalculate(this.transform.zoom);\n    },\n\n    _updateGlyphs: function() {\n        this.dispatcher.broadcast('set glyphs', this.style.stylesheet.glyphs);\n    },\n\n    _updateBuckets: function() {\n        // Transfer a stripped down version of the style to the workers. They only\n        // need the bucket information to know what features to extract from the tile.\n        this.dispatcher.broadcast('set buckets', this.style.orderedBuckets);\n\n        // clears all tiles to recalculate geometries (for changes to linecaps, linejoins, ...)\n        for (var s in this.sources) {\n            this.sources[s].load();\n        }\n\n        this.update();\n    }\n});\n\nutil.extendAll(Map.prototype, {\n\n    // debug code\n    _debug: false,\n    get debug() { return this._debug; },\n    set debug(value) { this._debug = value; this._rerender(); },\n\n    // continuous repaint\n    _repaint: false,\n    get repaint() { return this._repaint; },\n    set repaint(value) { this._repaint = value; this._rerender(); },\n\n    // polygon antialiasing\n    _antialiasing: true,\n    get antialiasing() { return this._antialiasing; },\n    set antialiasing(value) { this._antialiasing = value; this._rerender(); },\n\n    // show vertices\n    _vertices: false,\n    get vertices() { return this._vertices; },\n    set vertices(value) { this._vertices = value; this._rerender(); },\n\n    // show vertices\n    _loadNewTiles: true,\n    get loadNewTiles() { return this._loadNewTiles; },\n    set loadNewTiles(value) { this._loadNewTiles = value; this.update(); }\n});\n","'use strict';\n\nmodule.exports = Actor;\n\nfunction Actor(target, parent) {\n    this.target = target;\n    this.parent = parent;\n    this.callbacks = {};\n    this.callbackID = 0;\n    this.receive = this.receive.bind(this);\n    this.target.addEventListener('message', this.receive, false);\n}\n\nActor.prototype.receive = function(message) {\n    var data = message.data,\n        callback;\n\n    if (data.type == '<response>') {\n        callback = this.callbacks[data.id];\n        delete this.callbacks[data.id];\n        callback(data.error || null, data.data);\n    } else if (typeof data.id !== 'undefined') {\n        var id = data.id;\n        this.parent[data.type](data.data, function response(err, data, buffers) {\n            // console.warn('trying to clone', data, buffers, message.target);\n            message.target.postMessage({\n                type: '<response>',\n                id: String(id),\n                error: err ? String(err) : null,\n                data: data\n            }, buffers);\n        });\n    } else {\n        this.parent[data.type](data.data);\n    }\n};\n\nActor.prototype.send = function(type, data, callback, buffers) {\n    var id = null;\n    if (callback) this.callbacks[id = this.callbackID++] = callback;\n    this.target.postMessage({ type: type, id: String(id), data: data }, buffers);\n};\n","'use strict';\n\nexports.getJSON = function(url, callback) {\n    var xhr = new XMLHttpRequest();\n    xhr.open('GET', url, true);\n    xhr.onerror = function(e) {\n        callback(e);\n    };\n    xhr.onload = function() {\n        if (xhr.status >= 200 && xhr.status < 300 && xhr.response) {\n            var data;\n            try { data = JSON.parse(xhr.response); }\n            catch (err) { return callback(err); }\n            callback(null, data);\n        } else {\n            callback(new Error(xhr.statusText));\n        }\n    };\n    xhr.send();\n    return xhr;\n};\n\nexports.getArrayBuffer = function(url, callback) {\n    var xhr = new XMLHttpRequest();\n    xhr.open('GET', url, true);\n    xhr.responseType = 'arraybuffer';\n    xhr.onerror = function(e) {\n        callback(e);\n    };\n    xhr.onload = function() {\n        if (xhr.status >= 200 && xhr.status < 300 && xhr.response) {\n            callback(null, xhr.response);\n        } else {\n            callback(new Error(xhr.statusText));\n        }\n    };\n    xhr.send();\n    return xhr;\n};\n\nexports.getImage = function(url, callback) {\n    var img = new Image();\n    img.crossOrigin = 'Anonymous';\n    img.onload = function() {\n        callback(null, img);\n    };\n    img.src = url;\n    img.getData = function() { return getImageData(this); };\n    return img;\n};\n\nfunction getImageData(img) {\n    var canvas = document.createElement('canvas');\n    var context = canvas.getContext('2d');\n    canvas.width = img.width;\n    canvas.height = img.height;\n    context.drawImage(img, 0, 0);\n    return context.getImageData(0, 0, img.width, img.height).data;\n}\n","'use strict';\n\nvar frameName = (function() {\n    if (window.requestAnimationFrame) return 'requestAnimationFrame';\n    if (window.mozRequestAnimationFrame) return 'mozRequestAnimationFrame';\n    if (window.webkitRequestAnimationFrame) return 'webkitRequestAnimationFrame';\n    if (window.msRequestAnimationFrame) return 'msRequestAnimationFrame';\n})();\n\nexports.frame = function(fn) {\n    return window[frameName](fn);\n};\n\nexports.cancelFrame = function(id) {\n    (window.cancelRequestAnimationFrame ||\n        window.mozCancelRequestAnimationFrame ||\n        window.webkitCancelRequestAnimationFrame ||\n        window.msCancelRequestAnimationFrame)(id);\n};\n\nexports.timed = function (fn, dur, ctx) {\n    if (!dur) { return fn.call(ctx, 1); }\n\n    var abort = false,\n        start = window.performance ? window.performance.now() : Date.now();\n\n    function tick(now) {\n        if (abort) return;\n        if (!window.performance) now = Date.now();\n\n        if (now > start + dur) {\n            fn.call(ctx, 1);\n        } else {\n            fn.call(ctx, (now - start) / dur);\n            exports.frame(tick);\n        }\n    }\n\n    exports.frame(tick);\n\n    return function() { abort = true; };\n};\n\nexports.supported = function() {\n    var supports = [\n\n        function() { return typeof window !== 'undefined'; },\n\n        function() { return typeof document !== 'undefined'; },\n\n        function () {\n            return !!(Array.prototype &&\n                Array.prototype.every &&\n                Array.prototype.filter &&\n                Array.prototype.forEach &&\n                Array.prototype.indexOf &&\n                Array.prototype.lastIndexOf &&\n                Array.prototype.map &&\n                Array.prototype.some &&\n                Array.prototype.reduce &&\n                Array.prototype.reduceRight &&\n                Array.isArray);\n        },\n\n        function() {\n            return !!(Function.prototype && Function.prototype.bind),\n                !!(Object.keys &&\n                    Object.create &&\n                    Object.getPrototypeOf &&\n                    Object.getOwnPropertyNames &&\n                    Object.isSealed &&\n                    Object.isFrozen &&\n                    Object.isExtensible &&\n                    Object.getOwnPropertyDescriptor &&\n                    Object.defineProperty &&\n                    Object.defineProperties &&\n                    Object.seal &&\n                    Object.freeze &&\n                    Object.preventExtensions);\n        },\n\n        function() {\n            return 'JSON' in window && 'parse' in JSON && 'stringify' in JSON;\n        },\n\n        function() {\n            var canvas = document.createElement('canvas');\n            if ('supportsContext' in canvas) {\n                return canvas.supportsContext('webgl') || canvas.supportsContext('experimental-webgl');\n            }\n            return !!window.WebGLRenderingContext &&\n                (!!canvas.getContext('webgl') || !!canvas.getContext('experimental-webgl'));\n        },\n\n        function() { return 'Worker' in window; }\n    ];\n\n    for (var i = 0; i < supports.length; i++) {\n        if (!supports[i]()) return false;\n    }\n    return true;\n};\n\nexports.hardwareConcurrency = navigator.hardwareConcurrency || 8;\n\nObject.defineProperty(exports, 'devicePixelRatio', {\n    get: function() { return window.devicePixelRatio; }\n});\n","'use strict';\n\nmodule.exports = Canvas;\n\nfunction Canvas(parent, container) {\n    this.canvas = document.createElement('canvas');\n    this.canvas.style.position = 'absolute';\n    this.canvas.classList.add('mapboxgl-canvas');\n    this.canvas.addEventListener('webglcontextlost', parent._contextLost.bind(parent), false);\n    this.canvas.addEventListener('webglcontextrestored', parent._contextRestored.bind(parent), false);\n    container.appendChild(this.canvas);\n}\n\nCanvas.prototype.resize = function(width, height) {\n    var pixelRatio = window.devicePixelRatio || 1;\n\n    // Request the required canvas size taking the pixelratio into account.\n    this.canvas.width = pixelRatio * width;\n    this.canvas.height = pixelRatio * height;\n\n    // Maintain the same canvas size, potentially downscaling it for HiDPI displays\n    this.canvas.style.width = width + 'px';\n    this.canvas.style.height = height + 'px';\n};\n\nCanvas.prototype.getWebGLContext = function() {\n    return this.canvas.getContext(\"experimental-webgl\", {\n        antialias: false,\n        alpha: true,\n        stencil: true,\n        depth: false\n    });\n};\n","'use strict';\n\nvar Actor = require('../actor.js');\n\nvar scripts = document.getElementsByTagName(\"script\");\nvar workerFile = scripts[scripts.length - 1].getAttribute('src');\nvar absolute = workerFile.indexOf('http') !== -1;\n\n\n// Manages the WebWorkers\nmodule.exports = Dispatcher;\nfunction Dispatcher(length, parent) {\n    this.actors = [];\n    this.currentActor = 0;\n\n    var url, blob, i;\n\n    for (i = 0; i < length; i++) {\n        // due to cross domain issues we can't load it directly with the url,\n        // so create a blob and object url and load that\n        if (absolute) {\n            blob = new Blob(['importScripts(\"' + workerFile + '\");'], {type : 'application/javascript'});\n            url = window.URL.createObjectURL(blob);\n        } else {\n            url = workerFile;\n        }\n\n        var worker = new Worker(url);\n        var actor = new Actor(worker, parent);\n        actor.name = \"Worker \" + i;\n        this.actors.push(actor);\n    }\n}\n\nDispatcher.prototype.broadcast = function(type, data) {\n    for (var i = 0; i < this.actors.length; i++) {\n        this.actors[i].send(type, data);\n    }\n};\n\nDispatcher.prototype.send = function(type, data, callback, targetID, buffers) {\n    if (typeof targetID !== 'number' || isNaN(targetID)) {\n        // Use round robin to send requests to web workers.\n        targetID = this.currentActor = (this.currentActor + 1) % this.actors.length;\n    }\n\n    this.actors[targetID].send(type, data, callback, buffers);\n    return targetID;\n};\n\n","'use strict';\n\nmodule.exports = {\n    HTTP_URL: 'http://a.tiles.mapbox.com/v4',\n    HTTPS_URL: 'https://a.tiles.mapbox.com/v4',\n    FORCE_HTTPS: false,\n    REQUIRE_ACCESS_TOKEN: true\n};\n","'use strict';\n\nexports.create = function (tagName, className, container) {\n    var el = document.createElement(tagName);\n    if (className) el.className = className;\n    if (container) container.appendChild(el);\n    return el;\n};\n\nfunction preventDefault(e) {\n    e.preventDefault();\n}\n\nvar docEl = typeof document !== 'undefined' ? document.documentElement : {},\n    selectProp =\n        'userSelect' in docEl ? 'userSelect' :\n        'MozUserSelect' in docEl ? 'MozUserSelect' :\n        'WebkitUserSelect' in docEl ? 'WebkitUserSelect' : null,\n    userSelect;\n\nexports.disableDrag = function () {\n    window.addEventListener('dragstart', preventDefault);\n\n    if ('onselectstart' in document) window.addEventListener('selectstart', preventDefault);\n    else if (selectProp) {\n        userSelect = docEl.style[selectProp];\n        docEl.style[selectProp] = 'none';\n    }\n};\nexports.enableDrag = function () {\n    window.removeEventListener('dragstart', preventDefault);\n\n    if ('onselectstart' in document) window.removeEventListener('selectstart', preventDefault);\n    else if (selectProp) docEl.style[selectProp] = userSelect;\n};\n","'use strict';\n\nvar util = require('./util.js');\n\nmodule.exports = {\n    on: function(type, fn) {\n        this._events = this._events || {};\n        this._events[type] = this._events[type] || [];\n        this._events[type].push(fn);\n\n        return this;\n    },\n\n    off: function(type, fn) {\n        if (!type) {\n            // clear all listeners if no arguments specified\n            delete this._events;\n            return this;\n        }\n\n        if (!this.listens(type)) return this;\n\n        if (fn) {\n            var idx = this._events[type].indexOf(fn);\n            if (idx >= 0) {\n                this._events[type].splice(idx, 1);\n            }\n            if (!this._events[type].length) {\n                delete this._events[type];\n            }\n        } else {\n            delete this._events[type];\n        }\n\n        return this;\n    },\n\n    fire: function(type, data) {\n        if (!this.listens(type)) return this;\n\n        data = util.extend({}, data);\n        util.extend(data, {type: type, target: this});\n\n        // make sure adding/removing listeners inside other listeners won't cause infinite loop\n        var listeners = this._events[type].slice();\n\n        for (var i = 0; i < listeners.length; i++) {\n            listeners[i].call(this, data);\n        }\n\n        return this;\n    },\n\n    listens: function(type) {\n        return !!(this._events && this._events[type]);\n    }\n};\n","'use strict';\n\nmodule.exports = Glyphs;\nfunction Glyphs(buffer, end) {\n    // Public\n    this.stacks = {};\n    // Private\n    this._buffer = buffer;\n\n    var val, tag;\n    if (typeof end === 'undefined') end = buffer.length;\n    while (buffer.pos < end) {\n        val = buffer.readVarint();\n        tag = val >> 3;\n        if (tag == 1) {\n            var fontstack = this.readFontstack();\n            this.stacks[fontstack.name] = fontstack;\n        } else {\n            // console.warn('skipping tile tag ' + tag);\n            buffer.skip(val);\n        }\n    }\n}\n\nGlyphs.prototype.readFontstack = function() {\n    var buffer = this._buffer;\n    var fontstack = { glyphs: {} };\n\n    var bytes = buffer.readVarint();\n    var val, tag;\n    var end = buffer.pos + bytes;\n    while (buffer.pos < end) {\n        val = buffer.readVarint();\n        tag = val >> 3;\n\n        if (tag == 1) {\n            fontstack.name = buffer.readString();\n        } else if (tag == 2) {\n            var range = buffer.readString();\n            fontstack.range = range;\n        } else if (tag == 3) {\n            var glyph = this.readGlyph();\n            fontstack.glyphs[glyph.id] = glyph;\n        } else {\n            buffer.skip(val);\n        }\n    }\n\n    return fontstack;\n};\n\nGlyphs.prototype.readGlyph = function() {\n    var buffer = this._buffer;\n    var glyph = {};\n\n    var bytes = buffer.readVarint();\n    var val, tag;\n    var end = buffer.pos + bytes;\n    while (buffer.pos < end) {\n        val = buffer.readVarint();\n        tag = val >> 3;\n\n        if (tag == 1) {\n            glyph.id = buffer.readVarint();\n        } else if (tag == 2) {\n            glyph.bitmap = buffer.readBuffer();\n        } else if (tag == 3) {\n            glyph.width = buffer.readVarint();\n        } else if (tag == 4) {\n            glyph.height = buffer.readVarint();\n        } else if (tag == 5) {\n            glyph.left = buffer.readSVarint();\n        } else if (tag == 6) {\n            glyph.top = buffer.readSVarint();\n        } else if (tag == 7) {\n            glyph.advance = buffer.readVarint();\n        } else {\n            buffer.skip(val);\n        }\n    }\n\n    return glyph;\n};\n","'use strict';\n\n/*\n * A [most-recently-used cache](http://en.wikipedia.org/wiki/Cache_algorithms)\n * with hash lookup made possible by keeping a list of keys in parallel to\n * an array of dictionary of values\n */\nmodule.exports = MRUCache;\nfunction MRUCache(length, onRemove) {\n    this.max = length;\n    this.onRemove = onRemove;\n    this.reset();\n}\n\n/*\n * Clears the cache\n */\nMRUCache.prototype.reset = function() {\n    this.list = {};\n    this.order = [];\n\n    return this;\n};\n\n/*\n * Add a key, value combination to the cache, trimming its size if this pushes\n * it over max length.\n */\nMRUCache.prototype.add = function(key, data) {\n    this.list[key] = data;\n    this.order.push(key);\n\n    if (this.order.length > this.max) {\n        var removedData = this.get(this.order[0]);\n        if (removedData) this.onRemove(removedData);\n    }\n\n    return this;\n};\n\n/*\n * Determine whether the value attached to `key` is present\n */\nMRUCache.prototype.has = function(key) {\n    return key in this.list;\n};\n\n/*\n * List all keys in the cache\n */\nMRUCache.prototype.keys = function() {\n    return this.order;\n};\n\n/*\n * Get the value attached to a specific key. If the key is not found,\n * returns `null`\n */\nMRUCache.prototype.get = function(key) {\n    if (!this.has(key)) { return null; }\n\n    var data = this.list[key];\n\n    delete this.list[key];\n    this.order.splice(this.order.indexOf(key), 1);\n\n    return data;\n};\n","'use strict';\n\nmodule.exports = resolveTokens;\n\nvar tokenPattern = /{([\\w-]+)}/;\n\nfunction resolveTokens(properties, expression) {\n    var match;\n    var value;\n    var text = expression;\n    while ((match = text.match(tokenPattern))) {\n        value = typeof properties[match[1]] === 'undefined' ? '' : properties[match[1]];\n        text = text.replace(match[0], value);\n    }\n    return text;\n}\n","'use strict';\n\nvar config = require('./config');\n\nmodule.exports = function(path, accessToken) {\n    accessToken = accessToken || config.ACCESS_TOKEN;\n\n    if (!accessToken && config.REQUIRE_ACCESS_TOKEN) {\n        throw new Error('An API access token is required to use Mapbox GL. ' +\n            'See https://www.mapbox.com/developers/api/#access-tokens');\n    }\n\n    var url = ((typeof document !== 'undefined' && 'https:' === document.location.protocol) ||\n        config.FORCE_HTTPS) ? config.HTTPS_URL : config.HTTP_URL;\n    url += path;\n    url += url.indexOf('?') !== -1 ? '&access_token=' : '?access_token=';\n\n    if (config.REQUIRE_ACCESS_TOKEN) {\n        if (accessToken[0] === 's') {\n            throw new Error('Use a public access token (pk.*) with Mapbox GL JS, not a secret access token (sk.*). ' +\n                'See https://www.mapbox.com/developers/api/#access-tokens');\n        }\n\n        url += accessToken;\n    }\n\n    return url;\n};\n\nmodule.exports.tileJSON = function(mapID, accessToken) {\n    var url = module.exports('/' + mapID + '.json', accessToken);\n\n    // TileJSON requests need a secure flag appended to their URLs so\n    // that the server knows to send SSL-ified resource references.\n    if (url.indexOf('https') === 0)\n        url += '&secure';\n\n    return url;\n};\n","'use strict';\n\nvar UnitBezier = require('unitbezier');\n\nexports.easeCubicInOut = function (t) {\n    if (t <= 0) return 0;\n    if (t >= 1) return 1;\n    var t2 = t * t,\n        t3 = t2 * t;\n    return 4 * (t < 0.5 ? t3 : 3 * (t - t2) + t3 - 0.75);\n};\n\nexports.bezier = function(p1x, p1y, p2x, p2y) {\n    var bezier = new UnitBezier(p1x, p1y, p2x, p2y);\n    return function(t) {\n        return bezier.solve(t);\n    };\n};\n\nexports.ease = exports.bezier(0.25, 0.1, 0.25, 1);\n\nexports.interp = function (a, b, t) {\n    return (a * (1 - t)) + (b * t);\n};\n\nexports.premultiply = function (c) {\n    c[0] *= c[3];\n    c[1] *= c[3];\n    c[2] *= c[3];\n    return c;\n};\n\nexports.asyncEach = function (array, fn, callback) {\n    var remaining = array.length;\n    if (remaining === 0) return callback();\n    function check() { if (--remaining === 0) callback(); }\n    for (var i = 0; i < array.length; i++) fn(array[i], check);\n};\n\nexports.keysDifference = function (obj, other) {\n    var difference = [];\n    for (var i in obj) {\n        if (!(i in other)) {\n            difference.push(i);\n        }\n    }\n    return difference;\n};\n\nexports.extend = function (dest, src) {\n    for (var i in src) {\n        dest[i] = src[i];\n    }\n    return dest;\n};\n\nexports.extendAll = function (dest, src) {\n    for (var i in src) {\n        Object.defineProperty(dest, i, Object.getOwnPropertyDescriptor(src, i));\n    }\n    return dest;\n};\n\nexports.inherit = function (parent, props) {\n    var parentProto = typeof parent === 'function' ? parent.prototype : parent,\n        proto = Object.create(parentProto);\n    exports.extendAll(proto, props);\n    return proto;\n};\n\nvar id = 1;\n\nexports.uniqueId = function () {\n    return id++;\n};\n\nexports.throttle = function (fn, time, context) {\n    var lock, args, wrapperFn, later;\n\n    later = function () {\n        // reset lock and call if queued\n        lock = false;\n        if (args) {\n            wrapperFn.apply(context, args);\n            args = false;\n        }\n    };\n\n    wrapperFn = function () {\n        if (lock) {\n            // called too soon, queue to call later\n            args = arguments;\n\n        } else {\n            // call and lock until later\n            fn.apply(context, arguments);\n            setTimeout(later, time);\n            lock = true;\n        }\n    };\n\n    return wrapperFn;\n};\n\nexports.debounce = function(fn, time) {\n    var timer, args;\n\n    return function() {\n        args = arguments;\n        clearTimeout(timer);\n\n        timer = setTimeout(function() {\n            fn.apply(null, args);\n        }, time);\n    };\n};\n","/*!\n * The buffer module from node.js, for the browser.\n *\n * @author   Feross Aboukhadijeh <feross@feross.org> <http://feross.org>\n * @license  MIT\n */\n\nvar base64 = require('base64-js')\nvar ieee754 = require('ieee754')\n\nexports.Buffer = Buffer\nexports.SlowBuffer = Buffer\nexports.INSPECT_MAX_BYTES = 50\nBuffer.poolSize = 8192\n\n/**\n * If `TYPED_ARRAY_SUPPORT`:\n *   === true    Use Uint8Array implementation (fastest)\n *   === false   Use Object implementation (most compatible, even IE6)\n *\n * Browsers that support typed arrays are IE 10+, Firefox 4+, Chrome 7+, Safari 5.1+,\n * Opera 11.6+, iOS 4.2+.\n *\n * Note:\n *\n * - Implementation must support adding new properties to `Uint8Array` instances.\n *   Firefox 4-29 lacked support, fixed in Firefox 30+.\n *   See: https://bugzilla.mozilla.org/show_bug.cgi?id=695438.\n *\n *  - Chrome 9-10 is missing the `TypedArray.prototype.subarray` function.\n *\n *  - IE10 has a broken `TypedArray.prototype.subarray` function which returns arrays of\n *    incorrect length in some situations.\n *\n * We detect these buggy browsers and set `TYPED_ARRAY_SUPPORT` to `false` so they will\n * get the Object implementation, which is slower but will work correctly.\n */\nvar TYPED_ARRAY_SUPPORT = (function () {\n  try {\n    var buf = new ArrayBuffer(0)\n    var arr = new Uint8Array(buf)\n    arr.foo = function () { return 42 }\n    return 42 === arr.foo() && // typed array instances can be augmented\n        typeof arr.subarray === 'function' && // chrome 9-10 lack `subarray`\n        new Uint8Array(1).subarray(1, 1).byteLength === 0 // ie10 has broken `subarray`\n  } catch (e) {\n    return false\n  }\n})()\n\n/**\n * Class: Buffer\n * =============\n *\n * The Buffer constructor returns instances of `Uint8Array` that are augmented\n * with function properties for all the node `Buffer` API functions. We use\n * `Uint8Array` so that square bracket notation works as expected -- it returns\n * a single octet.\n *\n * By augmenting the instances, we can avoid modifying the `Uint8Array`\n * prototype.\n */\nfunction Buffer (subject, encoding, noZero) {\n  if (!(this instanceof Buffer))\n    return new Buffer(subject, encoding, noZero)\n\n  var type = typeof subject\n\n  // Find the length\n  var length\n  if (type === 'number')\n    length = subject > 0 ? subject >>> 0 : 0\n  else if (type === 'string') {\n    if (encoding === 'base64')\n      subject = base64clean(subject)\n    length = Buffer.byteLength(subject, encoding)\n  } else if (type === 'object' && subject !== null) { // assume object is array-like\n    if (subject.type === 'Buffer' && isArray(subject.data))\n      subject = subject.data\n    length = +subject.length > 0 ? Math.floor(+subject.length) : 0\n  } else\n    throw new Error('First argument needs to be a number, array or string.')\n\n  var buf\n  if (TYPED_ARRAY_SUPPORT) {\n    // Preferred: Return an augmented `Uint8Array` instance for best performance\n    buf = Buffer._augment(new Uint8Array(length))\n  } else {\n    // Fallback: Return THIS instance of Buffer (created by `new`)\n    buf = this\n    buf.length = length\n    buf._isBuffer = true\n  }\n\n  var i\n  if (TYPED_ARRAY_SUPPORT && typeof subject.byteLength === 'number') {\n    // Speed optimization -- use set if we're copying from a typed array\n    buf._set(subject)\n  } else if (isArrayish(subject)) {\n    // Treat array-ish objects as a byte array\n    if (Buffer.isBuffer(subject)) {\n      for (i = 0; i < length; i++)\n        buf[i] = subject.readUInt8(i)\n    } else {\n      for (i = 0; i < length; i++)\n        buf[i] = ((subject[i] % 256) + 256) % 256\n    }\n  } else if (type === 'string') {\n    buf.write(subject, 0, encoding)\n  } else if (type === 'number' && !TYPED_ARRAY_SUPPORT && !noZero) {\n    for (i = 0; i < length; i++) {\n      buf[i] = 0\n    }\n  }\n\n  return buf\n}\n\n// STATIC METHODS\n// ==============\n\nBuffer.isEncoding = function (encoding) {\n  switch (String(encoding).toLowerCase()) {\n    case 'hex':\n    case 'utf8':\n    case 'utf-8':\n    case 'ascii':\n    case 'binary':\n    case 'base64':\n    case 'raw':\n    case 'ucs2':\n    case 'ucs-2':\n    case 'utf16le':\n    case 'utf-16le':\n      return true\n    default:\n      return false\n  }\n}\n\nBuffer.isBuffer = function (b) {\n  return !!(b != null && b._isBuffer)\n}\n\nBuffer.byteLength = function (str, encoding) {\n  var ret\n  str = str.toString()\n  switch (encoding || 'utf8') {\n    case 'hex':\n      ret = str.length / 2\n      break\n    case 'utf8':\n    case 'utf-8':\n      ret = utf8ToBytes(str).length\n      break\n    case 'ascii':\n    case 'binary':\n    case 'raw':\n      ret = str.length\n      break\n    case 'base64':\n      ret = base64ToBytes(str).length\n      break\n    case 'ucs2':\n    case 'ucs-2':\n    case 'utf16le':\n    case 'utf-16le':\n      ret = str.length * 2\n      break\n    default:\n      throw new Error('Unknown encoding')\n  }\n  return ret\n}\n\nBuffer.concat = function (list, totalLength) {\n  assert(isArray(list), 'Usage: Buffer.concat(list[, length])')\n\n  if (list.length === 0) {\n    return new Buffer(0)\n  } else if (list.length === 1) {\n    return list[0]\n  }\n\n  var i\n  if (totalLength === undefined) {\n    totalLength = 0\n    for (i = 0; i < list.length; i++) {\n      totalLength += list[i].length\n    }\n  }\n\n  var buf = new Buffer(totalLength)\n  var pos = 0\n  for (i = 0; i < list.length; i++) {\n    var item = list[i]\n    item.copy(buf, pos)\n    pos += item.length\n  }\n  return buf\n}\n\nBuffer.compare = function (a, b) {\n  assert(Buffer.isBuffer(a) && Buffer.isBuffer(b), 'Arguments must be Buffers')\n  var x = a.length\n  var y = b.length\n  for (var i = 0, len = Math.min(x, y); i < len && a[i] === b[i]; i++) {}\n  if (i !== len) {\n    x = a[i]\n    y = b[i]\n  }\n  if (x < y) {\n    return -1\n  }\n  if (y < x) {\n    return 1\n  }\n  return 0\n}\n\n// BUFFER INSTANCE METHODS\n// =======================\n\nfunction hexWrite (buf, string, offset, length) {\n  offset = Number(offset) || 0\n  var remaining = buf.length - offset\n  if (!length) {\n    length = remaining\n  } else {\n    length = Number(length)\n    if (length > remaining) {\n      length = remaining\n    }\n  }\n\n  // must be an even number of digits\n  var strLen = string.length\n  assert(strLen % 2 === 0, 'Invalid hex string')\n\n  if (length > strLen / 2) {\n    length = strLen / 2\n  }\n  for (var i = 0; i < length; i++) {\n    var byte = parseInt(string.substr(i * 2, 2), 16)\n    assert(!isNaN(byte), 'Invalid hex string')\n    buf[offset + i] = byte\n  }\n  return i\n}\n\nfunction utf8Write (buf, string, offset, length) {\n  var charsWritten = blitBuffer(utf8ToBytes(string), buf, offset, length)\n  return charsWritten\n}\n\nfunction asciiWrite (buf, string, offset, length) {\n  var charsWritten = blitBuffer(asciiToBytes(string), buf, offset, length)\n  return charsWritten\n}\n\nfunction binaryWrite (buf, string, offset, length) {\n  return asciiWrite(buf, string, offset, length)\n}\n\nfunction base64Write (buf, string, offset, length) {\n  var charsWritten = blitBuffer(base64ToBytes(string), buf, offset, length)\n  return charsWritten\n}\n\nfunction utf16leWrite (buf, string, offset, length) {\n  var charsWritten = blitBuffer(utf16leToBytes(string), buf, offset, length)\n  return charsWritten\n}\n\nBuffer.prototype.write = function (string, offset, length, encoding) {\n  // Support both (string, offset, length, encoding)\n  // and the legacy (string, encoding, offset, length)\n  if (isFinite(offset)) {\n    if (!isFinite(length)) {\n      encoding = length\n      length = undefined\n    }\n  } else {  // legacy\n    var swap = encoding\n    encoding = offset\n    offset = length\n    length = swap\n  }\n\n  offset = Number(offset) || 0\n  var remaining = this.length - offset\n  if (!length) {\n    length = remaining\n  } else {\n    length = Number(length)\n    if (length > remaining) {\n      length = remaining\n    }\n  }\n  encoding = String(encoding || 'utf8').toLowerCase()\n\n  var ret\n  switch (encoding) {\n    case 'hex':\n      ret = hexWrite(this, string, offset, length)\n      break\n    case 'utf8':\n    case 'utf-8':\n      ret = utf8Write(this, string, offset, length)\n      break\n    case 'ascii':\n      ret = asciiWrite(this, string, offset, length)\n      break\n    case 'binary':\n      ret = binaryWrite(this, string, offset, length)\n      break\n    case 'base64':\n      ret = base64Write(this, string, offset, length)\n      break\n    case 'ucs2':\n    case 'ucs-2':\n    case 'utf16le':\n    case 'utf-16le':\n      ret = utf16leWrite(this, string, offset, length)\n      break\n    default:\n      throw new Error('Unknown encoding')\n  }\n  return ret\n}\n\nBuffer.prototype.toString = function (encoding, start, end) {\n  var self = this\n\n  encoding = String(encoding || 'utf8').toLowerCase()\n  start = Number(start) || 0\n  end = (end === undefined) ? self.length : Number(end)\n\n  // Fastpath empty strings\n  if (end === start)\n    return ''\n\n  var ret\n  switch (encoding) {\n    case 'hex':\n      ret = hexSlice(self, start, end)\n      break\n    case 'utf8':\n    case 'utf-8':\n      ret = utf8Slice(self, start, end)\n      break\n    case 'ascii':\n      ret = asciiSlice(self, start, end)\n      break\n    case 'binary':\n      ret = binarySlice(self, start, end)\n      break\n    case 'base64':\n      ret = base64Slice(self, start, end)\n      break\n    case 'ucs2':\n    case 'ucs-2':\n    case 'utf16le':\n    case 'utf-16le':\n      ret = utf16leSlice(self, start, end)\n      break\n    default:\n      throw new Error('Unknown encoding')\n  }\n  return ret\n}\n\nBuffer.prototype.toJSON = function () {\n  return {\n    type: 'Buffer',\n    data: Array.prototype.slice.call(this._arr || this, 0)\n  }\n}\n\nBuffer.prototype.equals = function (b) {\n  assert(Buffer.isBuffer(b), 'Argument must be a Buffer')\n  return Buffer.compare(this, b) === 0\n}\n\nBuffer.prototype.compare = function (b) {\n  assert(Buffer.isBuffer(b), 'Argument must be a Buffer')\n  return Buffer.compare(this, b)\n}\n\n// copy(targetBuffer, targetStart=0, sourceStart=0, sourceEnd=buffer.length)\nBuffer.prototype.copy = function (target, target_start, start, end) {\n  var source = this\n\n  if (!start) start = 0\n  if (!end && end !== 0) end = this.length\n  if (!target_start) target_start = 0\n\n  // Copy 0 bytes; we're done\n  if (end === start) return\n  if (target.length === 0 || source.length === 0) return\n\n  // Fatal error conditions\n  assert(end >= start, 'sourceEnd < sourceStart')\n  assert(target_start >= 0 && target_start < target.length,\n      'targetStart out of bounds')\n  assert(start >= 0 && start < source.length, 'sourceStart out of bounds')\n  assert(end >= 0 && end <= source.length, 'sourceEnd out of bounds')\n\n  // Are we oob?\n  if (end > this.length)\n    end = this.length\n  if (target.length - target_start < end - start)\n    end = target.length - target_start + start\n\n  var len = end - start\n\n  if (len < 100 || !TYPED_ARRAY_SUPPORT) {\n    for (var i = 0; i < len; i++) {\n      target[i + target_start] = this[i + start]\n    }\n  } else {\n    target._set(this.subarray(start, start + len), target_start)\n  }\n}\n\nfunction base64Slice (buf, start, end) {\n  if (start === 0 && end === buf.length) {\n    return base64.fromByteArray(buf)\n  } else {\n    return base64.fromByteArray(buf.slice(start, end))\n  }\n}\n\nfunction utf8Slice (buf, start, end) {\n  var res = ''\n  var tmp = ''\n  end = Math.min(buf.length, end)\n\n  for (var i = start; i < end; i++) {\n    if (buf[i] <= 0x7F) {\n      res += decodeUtf8Char(tmp) + String.fromCharCode(buf[i])\n      tmp = ''\n    } else {\n      tmp += '%' + buf[i].toString(16)\n    }\n  }\n\n  return res + decodeUtf8Char(tmp)\n}\n\nfunction asciiSlice (buf, start, end) {\n  var ret = ''\n  end = Math.min(buf.length, end)\n\n  for (var i = start; i < end; i++) {\n    ret += String.fromCharCode(buf[i])\n  }\n  return ret\n}\n\nfunction binarySlice (buf, start, end) {\n  return asciiSlice(buf, start, end)\n}\n\nfunction hexSlice (buf, start, end) {\n  var len = buf.length\n\n  if (!start || start < 0) start = 0\n  if (!end || end < 0 || end > len) end = len\n\n  var out = ''\n  for (var i = start; i < end; i++) {\n    out += toHex(buf[i])\n  }\n  return out\n}\n\nfunction utf16leSlice (buf, start, end) {\n  var bytes = buf.slice(start, end)\n  var res = ''\n  for (var i = 0; i < bytes.length; i += 2) {\n    res += String.fromCharCode(bytes[i] + bytes[i + 1] * 256)\n  }\n  return res\n}\n\nBuffer.prototype.slice = function (start, end) {\n  var len = this.length\n  start = ~~start\n  end = end === undefined ? len : ~~end\n\n  if (start < 0) {\n    start += len;\n    if (start < 0)\n      start = 0\n  } else if (start > len) {\n    start = len\n  }\n\n  if (end < 0) {\n    end += len\n    if (end < 0)\n      end = 0\n  } else if (end > len) {\n    end = len\n  }\n\n  if (end < start)\n    end = start\n\n  if (TYPED_ARRAY_SUPPORT) {\n    return Buffer._augment(this.subarray(start, end))\n  } else {\n    var sliceLen = end - start\n    var newBuf = new Buffer(sliceLen, undefined, true)\n    for (var i = 0; i < sliceLen; i++) {\n      newBuf[i] = this[i + start]\n    }\n    return newBuf\n  }\n}\n\n// `get` will be removed in Node 0.13+\nBuffer.prototype.get = function (offset) {\n  console.log('.get() is deprecated. Access using array indexes instead.')\n  return this.readUInt8(offset)\n}\n\n// `set` will be removed in Node 0.13+\nBuffer.prototype.set = function (v, offset) {\n  console.log('.set() is deprecated. Access using array indexes instead.')\n  return this.writeUInt8(v, offset)\n}\n\nBuffer.prototype.readUInt8 = function (offset, noAssert) {\n  if (!noAssert) {\n    assert(offset !== undefined && offset !== null, 'missing offset')\n    assert(offset < this.length, 'Trying to read beyond buffer length')\n  }\n\n  if (offset >= this.length)\n    return\n\n  return this[offset]\n}\n\nfunction readUInt16 (buf, offset, littleEndian, noAssert) {\n  if (!noAssert) {\n    assert(typeof littleEndian === 'boolean', 'missing or invalid endian')\n    assert(offset !== undefined && offset !== null, 'missing offset')\n    assert(offset + 1 < buf.length, 'Trying to read beyond buffer length')\n  }\n\n  var len = buf.length\n  if (offset >= len)\n    return\n\n  var val\n  if (littleEndian) {\n    val = buf[offset]\n    if (offset + 1 < len)\n      val |= buf[offset + 1] << 8\n  } else {\n    val = buf[offset] << 8\n    if (offset + 1 < len)\n      val |= buf[offset + 1]\n  }\n  return val\n}\n\nBuffer.prototype.readUInt16LE = function (offset, noAssert) {\n  return readUInt16(this, offset, true, noAssert)\n}\n\nBuffer.prototype.readUInt16BE = function (offset, noAssert) {\n  return readUInt16(this, offset, false, noAssert)\n}\n\nfunction readUInt32 (buf, offset, littleEndian, noAssert) {\n  if (!noAssert) {\n    assert(typeof littleEndian === 'boolean', 'missing or invalid endian')\n    assert(offset !== undefined && offset !== null, 'missing offset')\n    assert(offset + 3 < buf.length, 'Trying to read beyond buffer length')\n  }\n\n  var len = buf.length\n  if (offset >= len)\n    return\n\n  var val\n  if (littleEndian) {\n    if (offset + 2 < len)\n      val = buf[offset + 2] << 16\n    if (offset + 1 < len)\n      val |= buf[offset + 1] << 8\n    val |= buf[offset]\n    if (offset + 3 < len)\n      val = val + (buf[offset + 3] << 24 >>> 0)\n  } else {\n    if (offset + 1 < len)\n      val = buf[offset + 1] << 16\n    if (offset + 2 < len)\n      val |= buf[offset + 2] << 8\n    if (offset + 3 < len)\n      val |= buf[offset + 3]\n    val = val + (buf[offset] << 24 >>> 0)\n  }\n  return val\n}\n\nBuffer.prototype.readUInt32LE = function (offset, noAssert) {\n  return readUInt32(this, offset, true, noAssert)\n}\n\nBuffer.prototype.readUInt32BE = function (offset, noAssert) {\n  return readUInt32(this, offset, false, noAssert)\n}\n\nBuffer.prototype.readInt8 = function (offset, noAssert) {\n  if (!noAssert) {\n    assert(offset !== undefined && offset !== null,\n        'missing offset')\n    assert(offset < this.length, 'Trying to read beyond buffer length')\n  }\n\n  if (offset >= this.length)\n    return\n\n  var neg = this[offset] & 0x80\n  if (neg)\n    return (0xff - this[offset] + 1) * -1\n  else\n    return this[offset]\n}\n\nfunction readInt16 (buf, offset, littleEndian, noAssert) {\n  if (!noAssert) {\n    assert(typeof littleEndian === 'boolean', 'missing or invalid endian')\n    assert(offset !== undefined && offset !== null, 'missing offset')\n    assert(offset + 1 < buf.length, 'Trying to read beyond buffer length')\n  }\n\n  var len = buf.length\n  if (offset >= len)\n    return\n\n  var val = readUInt16(buf, offset, littleEndian, true)\n  var neg = val & 0x8000\n  if (neg)\n    return (0xffff - val + 1) * -1\n  else\n    return val\n}\n\nBuffer.prototype.readInt16LE = function (offset, noAssert) {\n  return readInt16(this, offset, true, noAssert)\n}\n\nBuffer.prototype.readInt16BE = function (offset, noAssert) {\n  return readInt16(this, offset, false, noAssert)\n}\n\nfunction readInt32 (buf, offset, littleEndian, noAssert) {\n  if (!noAssert) {\n    assert(typeof littleEndian === 'boolean', 'missing or invalid endian')\n    assert(offset !== undefined && offset !== null, 'missing offset')\n    assert(offset + 3 < buf.length, 'Trying to read beyond buffer length')\n  }\n\n  var len = buf.length\n  if (offset >= len)\n    return\n\n  var val = readUInt32(buf, offset, littleEndian, true)\n  var neg = val & 0x80000000\n  if (neg)\n    return (0xffffffff - val + 1) * -1\n  else\n    return val\n}\n\nBuffer.prototype.readInt32LE = function (offset, noAssert) {\n  return readInt32(this, offset, true, noAssert)\n}\n\nBuffer.prototype.readInt32BE = function (offset, noAssert) {\n  return readInt32(this, offset, false, noAssert)\n}\n\nfunction readFloat (buf, offset, littleEndian, noAssert) {\n  if (!noAssert) {\n    assert(typeof littleEndian === 'boolean', 'missing or invalid endian')\n    assert(offset + 3 < buf.length, 'Trying to read beyond buffer length')\n  }\n\n  return ieee754.read(buf, offset, littleEndian, 23, 4)\n}\n\nBuffer.prototype.readFloatLE = function (offset, noAssert) {\n  return readFloat(this, offset, true, noAssert)\n}\n\nBuffer.prototype.readFloatBE = function (offset, noAssert) {\n  return readFloat(this, offset, false, noAssert)\n}\n\nfunction readDouble (buf, offset, littleEndian, noAssert) {\n  if (!noAssert) {\n    assert(typeof littleEndian === 'boolean', 'missing or invalid endian')\n    assert(offset + 7 < buf.length, 'Trying to read beyond buffer length')\n  }\n\n  return ieee754.read(buf, offset, littleEndian, 52, 8)\n}\n\nBuffer.prototype.readDoubleLE = function (offset, noAssert) {\n  return readDouble(this, offset, true, noAssert)\n}\n\nBuffer.prototype.readDoubleBE = function (offset, noAssert) {\n  return readDouble(this, offset, false, noAssert)\n}\n\nBuffer.prototype.writeUInt8 = function (value, offset, noAssert) {\n  if (!noAssert) {\n    assert(value !== undefined && value !== null, 'missing value')\n    assert(offset !== undefined && offset !== null, 'missing offset')\n    assert(offset < this.length, 'trying to write beyond buffer length')\n    verifuint(value, 0xff)\n  }\n\n  if (offset >= this.length) return\n\n  this[offset] = value\n  return offset + 1\n}\n\nfunction writeUInt16 (buf, value, offset, littleEndian, noAssert) {\n  if (!noAssert) {\n    assert(value !== undefined && value !== null, 'missing value')\n    assert(typeof littleEndian === 'boolean', 'missing or invalid endian')\n    assert(offset !== undefined && offset !== null, 'missing offset')\n    assert(offset + 1 < buf.length, 'trying to write beyond buffer length')\n    verifuint(value, 0xffff)\n  }\n\n  var len = buf.length\n  if (offset >= len)\n    return\n\n  for (var i = 0, j = Math.min(len - offset, 2); i < j; i++) {\n    buf[offset + i] =\n        (value & (0xff << (8 * (littleEndian ? i : 1 - i)))) >>>\n            (littleEndian ? i : 1 - i) * 8\n  }\n  return offset + 2\n}\n\nBuffer.prototype.writeUInt16LE = function (value, offset, noAssert) {\n  return writeUInt16(this, value, offset, true, noAssert)\n}\n\nBuffer.prototype.writeUInt16BE = function (value, offset, noAssert) {\n  return writeUInt16(this, value, offset, false, noAssert)\n}\n\nfunction writeUInt32 (buf, value, offset, littleEndian, noAssert) {\n  if (!noAssert) {\n    assert(value !== undefined && value !== null, 'missing value')\n    assert(typeof littleEndian === 'boolean', 'missing or invalid endian')\n    assert(offset !== undefined && offset !== null, 'missing offset')\n    assert(offset + 3 < buf.length, 'trying to write beyond buffer length')\n    verifuint(value, 0xffffffff)\n  }\n\n  var len = buf.length\n  if (offset >= len)\n    return\n\n  for (var i = 0, j = Math.min(len - offset, 4); i < j; i++) {\n    buf[offset + i] =\n        (value >>> (littleEndian ? i : 3 - i) * 8) & 0xff\n  }\n  return offset + 4\n}\n\nBuffer.prototype.writeUInt32LE = function (value, offset, noAssert) {\n  return writeUInt32(this, value, offset, true, noAssert)\n}\n\nBuffer.prototype.writeUInt32BE = function (value, offset, noAssert) {\n  return writeUInt32(this, value, offset, false, noAssert)\n}\n\nBuffer.prototype.writeInt8 = function (value, offset, noAssert) {\n  if (!noAssert) {\n    assert(value !== undefined && value !== null, 'missing value')\n    assert(offset !== undefined && offset !== null, 'missing offset')\n    assert(offset < this.length, 'Trying to write beyond buffer length')\n    verifsint(value, 0x7f, -0x80)\n  }\n\n  if (offset >= this.length)\n    return\n\n  if (value >= 0)\n    this.writeUInt8(value, offset, noAssert)\n  else\n    this.writeUInt8(0xff + value + 1, offset, noAssert)\n  return offset + 1\n}\n\nfunction writeInt16 (buf, value, offset, littleEndian, noAssert) {\n  if (!noAssert) {\n    assert(value !== undefined && value !== null, 'missing value')\n    assert(typeof littleEndian === 'boolean', 'missing or invalid endian')\n    assert(offset !== undefined && offset !== null, 'missing offset')\n    assert(offset + 1 < buf.length, 'Trying to write beyond buffer length')\n    verifsint(value, 0x7fff, -0x8000)\n  }\n\n  var len = buf.length\n  if (offset >= len)\n    return\n\n  if (value >= 0)\n    writeUInt16(buf, value, offset, littleEndian, noAssert)\n  else\n    writeUInt16(buf, 0xffff + value + 1, offset, littleEndian, noAssert)\n  return offset + 2\n}\n\nBuffer.prototype.writeInt16LE = function (value, offset, noAssert) {\n  return writeInt16(this, value, offset, true, noAssert)\n}\n\nBuffer.prototype.writeInt16BE = function (value, offset, noAssert) {\n  return writeInt16(this, value, offset, false, noAssert)\n}\n\nfunction writeInt32 (buf, value, offset, littleEndian, noAssert) {\n  if (!noAssert) {\n    assert(value !== undefined && value !== null, 'missing value')\n    assert(typeof littleEndian === 'boolean', 'missing or invalid endian')\n    assert(offset !== undefined && offset !== null, 'missing offset')\n    assert(offset + 3 < buf.length, 'Trying to write beyond buffer length')\n    verifsint(value, 0x7fffffff, -0x80000000)\n  }\n\n  var len = buf.length\n  if (offset >= len)\n    return\n\n  if (value >= 0)\n    writeUInt32(buf, value, offset, littleEndian, noAssert)\n  else\n    writeUInt32(buf, 0xffffffff + value + 1, offset, littleEndian, noAssert)\n  return offset + 4\n}\n\nBuffer.prototype.writeInt32LE = function (value, offset, noAssert) {\n  return writeInt32(this, value, offset, true, noAssert)\n}\n\nBuffer.prototype.writeInt32BE = function (value, offset, noAssert) {\n  return writeInt32(this, value, offset, false, noAssert)\n}\n\nfunction writeFloat (buf, value, offset, littleEndian, noAssert) {\n  if (!noAssert) {\n    assert(value !== undefined && value !== null, 'missing value')\n    assert(typeof littleEndian === 'boolean', 'missing or invalid endian')\n    assert(offset !== undefined && offset !== null, 'missing offset')\n    assert(offset + 3 < buf.length, 'Trying to write beyond buffer length')\n    verifIEEE754(value, 3.4028234663852886e+38, -3.4028234663852886e+38)\n  }\n\n  var len = buf.length\n  if (offset >= len)\n    return\n\n  ieee754.write(buf, value, offset, littleEndian, 23, 4)\n  return offset + 4\n}\n\nBuffer.prototype.writeFloatLE = function (value, offset, noAssert) {\n  return writeFloat(this, value, offset, true, noAssert)\n}\n\nBuffer.prototype.writeFloatBE = function (value, offset, noAssert) {\n  return writeFloat(this, value, offset, false, noAssert)\n}\n\nfunction writeDouble (buf, value, offset, littleEndian, noAssert) {\n  if (!noAssert) {\n    assert(value !== undefined && value !== null, 'missing value')\n    assert(typeof littleEndian === 'boolean', 'missing or invalid endian')\n    assert(offset !== undefined && offset !== null, 'missing offset')\n    assert(offset + 7 < buf.length,\n        'Trying to write beyond buffer length')\n    verifIEEE754(value, 1.7976931348623157E+308, -1.7976931348623157E+308)\n  }\n\n  var len = buf.length\n  if (offset >= len)\n    return\n\n  ieee754.write(buf, value, offset, littleEndian, 52, 8)\n  return offset + 8\n}\n\nBuffer.prototype.writeDoubleLE = function (value, offset, noAssert) {\n  return writeDouble(this, value, offset, true, noAssert)\n}\n\nBuffer.prototype.writeDoubleBE = function (value, offset, noAssert) {\n  return writeDouble(this, value, offset, false, noAssert)\n}\n\n// fill(value, start=0, end=buffer.length)\nBuffer.prototype.fill = function (value, start, end) {\n  if (!value) value = 0\n  if (!start) start = 0\n  if (!end) end = this.length\n\n  assert(end >= start, 'end < start')\n\n  // Fill 0 bytes; we're done\n  if (end === start) return\n  if (this.length === 0) return\n\n  assert(start >= 0 && start < this.length, 'start out of bounds')\n  assert(end >= 0 && end <= this.length, 'end out of bounds')\n\n  var i\n  if (typeof value === 'number') {\n    for (i = start; i < end; i++) {\n      this[i] = value\n    }\n  } else {\n    var bytes = utf8ToBytes(value.toString())\n    var len = bytes.length\n    for (i = start; i < end; i++) {\n      this[i] = bytes[i % len]\n    }\n  }\n\n  return this\n}\n\nBuffer.prototype.inspect = function () {\n  var out = []\n  var len = this.length\n  for (var i = 0; i < len; i++) {\n    out[i] = toHex(this[i])\n    if (i === exports.INSPECT_MAX_BYTES) {\n      out[i + 1] = '...'\n      break\n    }\n  }\n  return '<Buffer ' + out.join(' ') + '>'\n}\n\n/**\n * Creates a new `ArrayBuffer` with the *copied* memory of the buffer instance.\n * Added in Node 0.12. Only available in browsers that support ArrayBuffer.\n */\nBuffer.prototype.toArrayBuffer = function () {\n  if (typeof Uint8Array !== 'undefined') {\n    if (TYPED_ARRAY_SUPPORT) {\n      return (new Buffer(this)).buffer\n    } else {\n      var buf = new Uint8Array(this.length)\n      for (var i = 0, len = buf.length; i < len; i += 1) {\n        buf[i] = this[i]\n      }\n      return buf.buffer\n    }\n  } else {\n    throw new Error('Buffer.toArrayBuffer not supported in this browser')\n  }\n}\n\n// HELPER FUNCTIONS\n// ================\n\nvar BP = Buffer.prototype\n\n/**\n * Augment a Uint8Array *instance* (not the Uint8Array class!) with Buffer methods\n */\nBuffer._augment = function (arr) {\n  arr._isBuffer = true\n\n  // save reference to original Uint8Array get/set methods before overwriting\n  arr._get = arr.get\n  arr._set = arr.set\n\n  // deprecated, will be removed in node 0.13+\n  arr.get = BP.get\n  arr.set = BP.set\n\n  arr.write = BP.write\n  arr.toString = BP.toString\n  arr.toLocaleString = BP.toString\n  arr.toJSON = BP.toJSON\n  arr.equals = BP.equals\n  arr.compare = BP.compare\n  arr.copy = BP.copy\n  arr.slice = BP.slice\n  arr.readUInt8 = BP.readUInt8\n  arr.readUInt16LE = BP.readUInt16LE\n  arr.readUInt16BE = BP.readUInt16BE\n  arr.readUInt32LE = BP.readUInt32LE\n  arr.readUInt32BE = BP.readUInt32BE\n  arr.readInt8 = BP.readInt8\n  arr.readInt16LE = BP.readInt16LE\n  arr.readInt16BE = BP.readInt16BE\n  arr.readInt32LE = BP.readInt32LE\n  arr.readInt32BE = BP.readInt32BE\n  arr.readFloatLE = BP.readFloatLE\n  arr.readFloatBE = BP.readFloatBE\n  arr.readDoubleLE = BP.readDoubleLE\n  arr.readDoubleBE = BP.readDoubleBE\n  arr.writeUInt8 = BP.writeUInt8\n  arr.writeUInt16LE = BP.writeUInt16LE\n  arr.writeUInt16BE = BP.writeUInt16BE\n  arr.writeUInt32LE = BP.writeUInt32LE\n  arr.writeUInt32BE = BP.writeUInt32BE\n  arr.writeInt8 = BP.writeInt8\n  arr.writeInt16LE = BP.writeInt16LE\n  arr.writeInt16BE = BP.writeInt16BE\n  arr.writeInt32LE = BP.writeInt32LE\n  arr.writeInt32BE = BP.writeInt32BE\n  arr.writeFloatLE = BP.writeFloatLE\n  arr.writeFloatBE = BP.writeFloatBE\n  arr.writeDoubleLE = BP.writeDoubleLE\n  arr.writeDoubleBE = BP.writeDoubleBE\n  arr.fill = BP.fill\n  arr.inspect = BP.inspect\n  arr.toArrayBuffer = BP.toArrayBuffer\n\n  return arr\n}\n\nvar INVALID_BASE64_RE = /[^+\\/0-9A-z]/g\n\nfunction base64clean (str) {\n  // Node strips out invalid characters like \\n and \\t from the string, base64-js does not\n  str = stringtrim(str).replace(INVALID_BASE64_RE, '')\n  // Node allows for non-padded base64 strings (missing trailing ===), base64-js does not\n  while (str.length % 4 !== 0) {\n    str = str + '='\n  }\n  return str\n}\n\nfunction stringtrim (str) {\n  if (str.trim) return str.trim()\n  return str.replace(/^\\s+|\\s+$/g, '')\n}\n\nfunction isArray (subject) {\n  return (Array.isArray || function (subject) {\n    return Object.prototype.toString.call(subject) === '[object Array]'\n  })(subject)\n}\n\nfunction isArrayish (subject) {\n  return isArray(subject) || Buffer.isBuffer(subject) ||\n      subject && typeof subject === 'object' &&\n      typeof subject.length === 'number'\n}\n\nfunction toHex (n) {\n  if (n < 16) return '0' + n.toString(16)\n  return n.toString(16)\n}\n\nfunction utf8ToBytes (str) {\n  var byteArray = []\n  for (var i = 0; i < str.length; i++) {\n    var b = str.charCodeAt(i)\n    if (b <= 0x7F) {\n      byteArray.push(b)\n    } else {\n      var start = i\n      if (b >= 0xD800 && b <= 0xDFFF) i++\n      var h = encodeURIComponent(str.slice(start, i+1)).substr(1).split('%')\n      for (var j = 0; j < h.length; j++) {\n        byteArray.push(parseInt(h[j], 16))\n      }\n    }\n  }\n  return byteArray\n}\n\nfunction asciiToBytes (str) {\n  var byteArray = []\n  for (var i = 0; i < str.length; i++) {\n    // Node's code seems to be doing this and not & 0x7F..\n    byteArray.push(str.charCodeAt(i) & 0xFF)\n  }\n  return byteArray\n}\n\nfunction utf16leToBytes (str) {\n  var c, hi, lo\n  var byteArray = []\n  for (var i = 0; i < str.length; i++) {\n    c = str.charCodeAt(i)\n    hi = c >> 8\n    lo = c % 256\n    byteArray.push(lo)\n    byteArray.push(hi)\n  }\n\n  return byteArray\n}\n\nfunction base64ToBytes (str) {\n  return base64.toByteArray(str)\n}\n\nfunction blitBuffer (src, dst, offset, length) {\n  for (var i = 0; i < length; i++) {\n    if ((i + offset >= dst.length) || (i >= src.length))\n      break\n    dst[i + offset] = src[i]\n  }\n  return i\n}\n\nfunction decodeUtf8Char (str) {\n  try {\n    return decodeURIComponent(str)\n  } catch (err) {\n    return String.fromCharCode(0xFFFD) // UTF 8 invalid char\n  }\n}\n\n/*\n * We have to make sure that the value is a valid integer. This means that it\n * is non-negative. It has no fractional component and that it does not\n * exceed the maximum allowed value.\n */\nfunction verifuint (value, max) {\n  assert(typeof value === 'number', 'cannot write a non-number as a number')\n  assert(value >= 0, 'specified a negative value for writing an unsigned value')\n  assert(value <= max, 'value is larger than maximum value for type')\n  assert(Math.floor(value) === value, 'value has a fractional component')\n}\n\nfunction verifsint (value, max, min) {\n  assert(typeof value === 'number', 'cannot write a non-number as a number')\n  assert(value <= max, 'value larger than maximum allowed value')\n  assert(value >= min, 'value smaller than minimum allowed value')\n  assert(Math.floor(value) === value, 'value has a fractional component')\n}\n\nfunction verifIEEE754 (value, max, min) {\n  assert(typeof value === 'number', 'cannot write a non-number as a number')\n  assert(value <= max, 'value larger than maximum allowed value')\n  assert(value >= min, 'value smaller than minimum allowed value')\n}\n\nfunction assert (test, message) {\n  if (!test) throw new Error(message || 'Failed assertion')\n}\n","var lookup = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';\n\n;(function (exports) {\n\t'use strict';\n\n  var Arr = (typeof Uint8Array !== 'undefined')\n    ? Uint8Array\n    : Array\n\n\tvar PLUS   = '+'.charCodeAt(0)\n\tvar SLASH  = '/'.charCodeAt(0)\n\tvar NUMBER = '0'.charCodeAt(0)\n\tvar LOWER  = 'a'.charCodeAt(0)\n\tvar UPPER  = 'A'.charCodeAt(0)\n\n\tfunction decode (elt) {\n\t\tvar code = elt.charCodeAt(0)\n\t\tif (code === PLUS)\n\t\t\treturn 62 // '+'\n\t\tif (code === SLASH)\n\t\t\treturn 63 // '/'\n\t\tif (code < NUMBER)\n\t\t\treturn -1 //no match\n\t\tif (code < NUMBER + 10)\n\t\t\treturn code - NUMBER + 26 + 26\n\t\tif (code < UPPER + 26)\n\t\t\treturn code - UPPER\n\t\tif (code < LOWER + 26)\n\t\t\treturn code - LOWER + 26\n\t}\n\n\tfunction b64ToByteArray (b64) {\n\t\tvar i, j, l, tmp, placeHolders, arr\n\n\t\tif (b64.length % 4 > 0) {\n\t\t\tthrow new Error('Invalid string. Length must be a multiple of 4')\n\t\t}\n\n\t\t// the number of equal signs (place holders)\n\t\t// if there are two placeholders, than the two characters before it\n\t\t// represent one byte\n\t\t// if there is only one, then the three characters before it represent 2 bytes\n\t\t// this is just a cheap hack to not do indexOf twice\n\t\tvar len = b64.length\n\t\tplaceHolders = '=' === b64.charAt(len - 2) ? 2 : '=' === b64.charAt(len - 1) ? 1 : 0\n\n\t\t// base64 is 4/3 + up to two characters of the original data\n\t\tarr = new Arr(b64.length * 3 / 4 - placeHolders)\n\n\t\t// if there are placeholders, only get up to the last complete 4 chars\n\t\tl = placeHolders > 0 ? b64.length - 4 : b64.length\n\n\t\tvar L = 0\n\n\t\tfunction push (v) {\n\t\t\tarr[L++] = v\n\t\t}\n\n\t\tfor (i = 0, j = 0; i < l; i += 4, j += 3) {\n\t\t\ttmp = (decode(b64.charAt(i)) << 18) | (decode(b64.charAt(i + 1)) << 12) | (decode(b64.charAt(i + 2)) << 6) | decode(b64.charAt(i + 3))\n\t\t\tpush((tmp & 0xFF0000) >> 16)\n\t\t\tpush((tmp & 0xFF00) >> 8)\n\t\t\tpush(tmp & 0xFF)\n\t\t}\n\n\t\tif (placeHolders === 2) {\n\t\t\ttmp = (decode(b64.charAt(i)) << 2) | (decode(b64.charAt(i + 1)) >> 4)\n\t\t\tpush(tmp & 0xFF)\n\t\t} else if (placeHolders === 1) {\n\t\t\ttmp = (decode(b64.charAt(i)) << 10) | (decode(b64.charAt(i + 1)) << 4) | (decode(b64.charAt(i + 2)) >> 2)\n\t\t\tpush((tmp >> 8) & 0xFF)\n\t\t\tpush(tmp & 0xFF)\n\t\t}\n\n\t\treturn arr\n\t}\n\n\tfunction uint8ToBase64 (uint8) {\n\t\tvar i,\n\t\t\textraBytes = uint8.length % 3, // if we have 1 byte left, pad 2 bytes\n\t\t\toutput = \"\",\n\t\t\ttemp, length\n\n\t\tfunction encode (num) {\n\t\t\treturn lookup.charAt(num)\n\t\t}\n\n\t\tfunction tripletToBase64 (num) {\n\t\t\treturn encode(num >> 18 & 0x3F) + encode(num >> 12 & 0x3F) + encode(num >> 6 & 0x3F) + encode(num & 0x3F)\n\t\t}\n\n\t\t// go through the array every three bytes, we'll deal with trailing stuff later\n\t\tfor (i = 0, length = uint8.length - extraBytes; i < length; i += 3) {\n\t\t\ttemp = (uint8[i] << 16) + (uint8[i + 1] << 8) + (uint8[i + 2])\n\t\t\toutput += tripletToBase64(temp)\n\t\t}\n\n\t\t// pad the end with zeros, but make sure to not forget the extra bytes\n\t\tswitch (extraBytes) {\n\t\t\tcase 1:\n\t\t\t\ttemp = uint8[uint8.length - 1]\n\t\t\t\toutput += encode(temp >> 2)\n\t\t\t\toutput += encode((temp << 4) & 0x3F)\n\t\t\t\toutput += '=='\n\t\t\t\tbreak\n\t\t\tcase 2:\n\t\t\t\ttemp = (uint8[uint8.length - 2] << 8) + (uint8[uint8.length - 1])\n\t\t\t\toutput += encode(temp >> 10)\n\t\t\t\toutput += encode((temp >> 4) & 0x3F)\n\t\t\t\toutput += encode((temp << 2) & 0x3F)\n\t\t\t\toutput += '='\n\t\t\t\tbreak\n\t\t}\n\n\t\treturn output\n\t}\n\n\texports.toByteArray = b64ToByteArray\n\texports.fromByteArray = uint8ToBase64\n}(typeof exports === 'undefined' ? (this.base64js = {}) : exports))\n","exports.read = function(buffer, offset, isLE, mLen, nBytes) {\n  var e, m,\n      eLen = nBytes * 8 - mLen - 1,\n      eMax = (1 << eLen) - 1,\n      eBias = eMax >> 1,\n      nBits = -7,\n      i = isLE ? (nBytes - 1) : 0,\n      d = isLE ? -1 : 1,\n      s = buffer[offset + i];\n\n  i += d;\n\n  e = s & ((1 << (-nBits)) - 1);\n  s >>= (-nBits);\n  nBits += eLen;\n  for (; nBits > 0; e = e * 256 + buffer[offset + i], i += d, nBits -= 8);\n\n  m = e & ((1 << (-nBits)) - 1);\n  e >>= (-nBits);\n  nBits += mLen;\n  for (; nBits > 0; m = m * 256 + buffer[offset + i], i += d, nBits -= 8);\n\n  if (e === 0) {\n    e = 1 - eBias;\n  } else if (e === eMax) {\n    return m ? NaN : ((s ? -1 : 1) * Infinity);\n  } else {\n    m = m + Math.pow(2, mLen);\n    e = e - eBias;\n  }\n  return (s ? -1 : 1) * m * Math.pow(2, e - mLen);\n};\n\nexports.write = function(buffer, value, offset, isLE, mLen, nBytes) {\n  var e, m, c,\n      eLen = nBytes * 8 - mLen - 1,\n      eMax = (1 << eLen) - 1,\n      eBias = eMax >> 1,\n      rt = (mLen === 23 ? Math.pow(2, -24) - Math.pow(2, -77) : 0),\n      i = isLE ? 0 : (nBytes - 1),\n      d = isLE ? 1 : -1,\n      s = value < 0 || (value === 0 && 1 / value < 0) ? 1 : 0;\n\n  value = Math.abs(value);\n\n  if (isNaN(value) || value === Infinity) {\n    m = isNaN(value) ? 1 : 0;\n    e = eMax;\n  } else {\n    e = Math.floor(Math.log(value) / Math.LN2);\n    if (value * (c = Math.pow(2, -e)) < 1) {\n      e--;\n      c *= 2;\n    }\n    if (e + eBias >= 1) {\n      value += rt / c;\n    } else {\n      value += rt * Math.pow(2, 1 - eBias);\n    }\n    if (value * c >= 2) {\n      e++;\n      c /= 2;\n    }\n\n    if (e + eBias >= eMax) {\n      m = 0;\n      e = eMax;\n    } else if (e + eBias >= 1) {\n      m = (value * c - 1) * Math.pow(2, mLen);\n      e = e + eBias;\n    } else {\n      m = value * Math.pow(2, eBias - 1) * Math.pow(2, mLen);\n      e = 0;\n    }\n  }\n\n  for (; mLen >= 8; buffer[offset + i] = m & 0xff, i += d, m /= 256, mLen -= 8);\n\n  e = (e << mLen) | m;\n  eLen += mLen;\n  for (; eLen > 0; buffer[offset + i] = e & 0xff, i += d, e /= 256, eLen -= 8);\n\n  buffer[offset + i - d] |= s * 128;\n};\n","// (c) Dean McNamee <dean@gmail.com>, 2012.\n//\n// https://github.com/deanm/css-color-parser-js\n//\n// Permission is hereby granted, free of charge, to any person obtaining a copy\n// of this software and associated documentation files (the \"Software\"), to\n// deal in the Software without restriction, including without limitation the\n// rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n// sell copies of the Software, and to permit persons to whom the Software is\n// furnished to do so, subject to the following conditions:\n//\n// The above copyright notice and this permission notice shall be included in\n// all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n// IN THE SOFTWARE.\n\n// http://www.w3.org/TR/css3-color/\nvar kCSSColorTable = {\n  \"transparent\": [0,0,0,0], \"aliceblue\": [240,248,255,1],\n  \"antiquewhite\": [250,235,215,1], \"aqua\": [0,255,255,1],\n  \"aquamarine\": [127,255,212,1], \"azure\": [240,255,255,1],\n  \"beige\": [245,245,220,1], \"bisque\": [255,228,196,1],\n  \"black\": [0,0,0,1], \"blanchedalmond\": [255,235,205,1],\n  \"blue\": [0,0,255,1], \"blueviolet\": [138,43,226,1],\n  \"brown\": [165,42,42,1], \"burlywood\": [222,184,135,1],\n  \"cadetblue\": [95,158,160,1], \"chartreuse\": [127,255,0,1],\n  \"chocolate\": [210,105,30,1], \"coral\": [255,127,80,1],\n  \"cornflowerblue\": [100,149,237,1], \"cornsilk\": [255,248,220,1],\n  \"crimson\": [220,20,60,1], \"cyan\": [0,255,255,1],\n  \"darkblue\": [0,0,139,1], \"darkcyan\": [0,139,139,1],\n  \"darkgoldenrod\": [184,134,11,1], \"darkgray\": [169,169,169,1],\n  \"darkgreen\": [0,100,0,1], \"darkgrey\": [169,169,169,1],\n  \"darkkhaki\": [189,183,107,1], \"darkmagenta\": [139,0,139,1],\n  \"darkolivegreen\": [85,107,47,1], \"darkorange\": [255,140,0,1],\n  \"darkorchid\": [153,50,204,1], \"darkred\": [139,0,0,1],\n  \"darksalmon\": [233,150,122,1], \"darkseagreen\": [143,188,143,1],\n  \"darkslateblue\": [72,61,139,1], \"darkslategray\": [47,79,79,1],\n  \"darkslategrey\": [47,79,79,1], \"darkturquoise\": [0,206,209,1],\n  \"darkviolet\": [148,0,211,1], \"deeppink\": [255,20,147,1],\n  \"deepskyblue\": [0,191,255,1], \"dimgray\": [105,105,105,1],\n  \"dimgrey\": [105,105,105,1], \"dodgerblue\": [30,144,255,1],\n  \"firebrick\": [178,34,34,1], \"floralwhite\": [255,250,240,1],\n  \"forestgreen\": [34,139,34,1], \"fuchsia\": [255,0,255,1],\n  \"gainsboro\": [220,220,220,1], \"ghostwhite\": [248,248,255,1],\n  \"gold\": [255,215,0,1], \"goldenrod\": [218,165,32,1],\n  \"gray\": [128,128,128,1], \"green\": [0,128,0,1],\n  \"greenyellow\": [173,255,47,1], \"grey\": [128,128,128,1],\n  \"honeydew\": [240,255,240,1], \"hotpink\": [255,105,180,1],\n  \"indianred\": [205,92,92,1], \"indigo\": [75,0,130,1],\n  \"ivory\": [255,255,240,1], \"khaki\": [240,230,140,1],\n  \"lavender\": [230,230,250,1], \"lavenderblush\": [255,240,245,1],\n  \"lawngreen\": [124,252,0,1], \"lemonchiffon\": [255,250,205,1],\n  \"lightblue\": [173,216,230,1], \"lightcoral\": [240,128,128,1],\n  \"lightcyan\": [224,255,255,1], \"lightgoldenrodyellow\": [250,250,210,1],\n  \"lightgray\": [211,211,211,1], \"lightgreen\": [144,238,144,1],\n  \"lightgrey\": [211,211,211,1], \"lightpink\": [255,182,193,1],\n  \"lightsalmon\": [255,160,122,1], \"lightseagreen\": [32,178,170,1],\n  \"lightskyblue\": [135,206,250,1], \"lightslategray\": [119,136,153,1],\n  \"lightslategrey\": [119,136,153,1], \"lightsteelblue\": [176,196,222,1],\n  \"lightyellow\": [255,255,224,1], \"lime\": [0,255,0,1],\n  \"limegreen\": [50,205,50,1], \"linen\": [250,240,230,1],\n  \"magenta\": [255,0,255,1], \"maroon\": [128,0,0,1],\n  \"mediumaquamarine\": [102,205,170,1], \"mediumblue\": [0,0,205,1],\n  \"mediumorchid\": [186,85,211,1], \"mediumpurple\": [147,112,219,1],\n  \"mediumseagreen\": [60,179,113,1], \"mediumslateblue\": [123,104,238,1],\n  \"mediumspringgreen\": [0,250,154,1], \"mediumturquoise\": [72,209,204,1],\n  \"mediumvioletred\": [199,21,133,1], \"midnightblue\": [25,25,112,1],\n  \"mintcream\": [245,255,250,1], \"mistyrose\": [255,228,225,1],\n  \"moccasin\": [255,228,181,1], \"navajowhite\": [255,222,173,1],\n  \"navy\": [0,0,128,1], \"oldlace\": [253,245,230,1],\n  \"olive\": [128,128,0,1], \"olivedrab\": [107,142,35,1],\n  \"orange\": [255,165,0,1], \"orangered\": [255,69,0,1],\n  \"orchid\": [218,112,214,1], \"palegoldenrod\": [238,232,170,1],\n  \"palegreen\": [152,251,152,1], \"paleturquoise\": [175,238,238,1],\n  \"palevioletred\": [219,112,147,1], \"papayawhip\": [255,239,213,1],\n  \"peachpuff\": [255,218,185,1], \"peru\": [205,133,63,1],\n  \"pink\": [255,192,203,1], \"plum\": [221,160,221,1],\n  \"powderblue\": [176,224,230,1], \"purple\": [128,0,128,1],\n  \"red\": [255,0,0,1], \"rosybrown\": [188,143,143,1],\n  \"royalblue\": [65,105,225,1], \"saddlebrown\": [139,69,19,1],\n  \"salmon\": [250,128,114,1], \"sandybrown\": [244,164,96,1],\n  \"seagreen\": [46,139,87,1], \"seashell\": [255,245,238,1],\n  \"sienna\": [160,82,45,1], \"silver\": [192,192,192,1],\n  \"skyblue\": [135,206,235,1], \"slateblue\": [106,90,205,1],\n  \"slategray\": [112,128,144,1], \"slategrey\": [112,128,144,1],\n  \"snow\": [255,250,250,1], \"springgreen\": [0,255,127,1],\n  \"steelblue\": [70,130,180,1], \"tan\": [210,180,140,1],\n  \"teal\": [0,128,128,1], \"thistle\": [216,191,216,1],\n  \"tomato\": [255,99,71,1], \"turquoise\": [64,224,208,1],\n  \"violet\": [238,130,238,1], \"wheat\": [245,222,179,1],\n  \"white\": [255,255,255,1], \"whitesmoke\": [245,245,245,1],\n  \"yellow\": [255,255,0,1], \"yellowgreen\": [154,205,50,1]}\n\nfunction clamp_css_byte(i) {  // Clamp to integer 0 .. 255.\n  i = Math.round(i);  // Seems to be what Chrome does (vs truncation).\n  return i < 0 ? 0 : i > 255 ? 255 : i;\n}\n\nfunction clamp_css_float(f) {  // Clamp to float 0.0 .. 1.0.\n  return f < 0 ? 0 : f > 1 ? 1 : f;\n}\n\nfunction parse_css_int(str) {  // int or percentage.\n  if (str[str.length - 1] === '%')\n    return clamp_css_byte(parseFloat(str) / 100 * 255);\n  return clamp_css_byte(parseInt(str));\n}\n\nfunction parse_css_float(str) {  // float or percentage.\n  if (str[str.length - 1] === '%')\n    return clamp_css_float(parseFloat(str) / 100);\n  return clamp_css_float(parseFloat(str));\n}\n\nfunction css_hue_to_rgb(m1, m2, h) {\n  if (h < 0) h += 1;\n  else if (h > 1) h -= 1;\n\n  if (h * 6 < 1) return m1 + (m2 - m1) * h * 6;\n  if (h * 2 < 1) return m2;\n  if (h * 3 < 2) return m1 + (m2 - m1) * (2/3 - h) * 6;\n  return m1;\n}\n\nfunction parseCSSColor(css_str) {\n  // Remove all whitespace, not compliant, but should just be more accepting.\n  var str = css_str.replace(/ /g, '').toLowerCase();\n\n  // Color keywords (and transparent) lookup.\n  if (str in kCSSColorTable) return kCSSColorTable[str].slice();  // dup.\n\n  // #abc and #abc123 syntax.\n  if (str[0] === '#') {\n    if (str.length === 4) {\n      var iv = parseInt(str.substr(1), 16);  // TODO(deanm): Stricter parsing.\n      if (!(iv >= 0 && iv <= 0xfff)) return null;  // Covers NaN.\n      return [((iv & 0xf00) >> 4) | ((iv & 0xf00) >> 8),\n              (iv & 0xf0) | ((iv & 0xf0) >> 4),\n              (iv & 0xf) | ((iv & 0xf) << 4),\n              1];\n    } else if (str.length === 7) {\n      var iv = parseInt(str.substr(1), 16);  // TODO(deanm): Stricter parsing.\n      if (!(iv >= 0 && iv <= 0xffffff)) return null;  // Covers NaN.\n      return [(iv & 0xff0000) >> 16,\n              (iv & 0xff00) >> 8,\n              iv & 0xff,\n              1];\n    }\n\n    return null;\n  }\n\n  var op = str.indexOf('('), ep = str.indexOf(')');\n  if (op !== -1 && ep + 1 === str.length) {\n    var fname = str.substr(0, op);\n    var params = str.substr(op+1, ep-(op+1)).split(',');\n    var alpha = 1;  // To allow case fallthrough.\n    switch (fname) {\n      case 'rgba':\n        if (params.length !== 4) return null;\n        alpha = parse_css_float(params.pop());\n        // Fall through.\n      case 'rgb':\n        if (params.length !== 3) return null;\n        return [parse_css_int(params[0]),\n                parse_css_int(params[1]),\n                parse_css_int(params[2]),\n                alpha];\n      case 'hsla':\n        if (params.length !== 4) return null;\n        alpha = parse_css_float(params.pop());\n        // Fall through.\n      case 'hsl':\n        if (params.length !== 3) return null;\n        var h = (((parseFloat(params[0]) % 360) + 360) % 360) / 360;  // 0 .. 1\n        // NOTE(deanm): According to the CSS spec s/l should only be\n        // percentages, but we don't bother and let float or percentage.\n        var s = parse_css_float(params[1]);\n        var l = parse_css_float(params[2]);\n        var m2 = l <= 0.5 ? l * (s + 1) : l + s - l * s;\n        var m1 = l * 2 - m2;\n        return [clamp_css_byte(css_hue_to_rgb(m1, m2, h+1/3) * 255),\n                clamp_css_byte(css_hue_to_rgb(m1, m2, h) * 255),\n                clamp_css_byte(css_hue_to_rgb(m1, m2, h-1/3) * 255),\n                alpha];\n      default:\n        return null;\n    }\n  }\n\n  return null;\n}\n\ntry { exports.parseCSSColor = parseCSSColor } catch(e) { }\n","var geojsonArea = require('geojson-area');\n\nmodule.exports = rewind;\n\nfunction rewind(gj, outer) {\n    switch ((gj && gj.type) || null) {\n        case 'FeatureCollection':\n            gj.features = gj.features.map(curryOuter(rewind, outer));\n            return gj;\n        case 'Feature':\n            gj.geometry = rewind(gj.geometry, outer);\n            return gj;\n        case 'Polygon':\n        case 'MultiPolygon':\n            return correct(gj, outer);\n        default:\n            return gj;\n    }\n}\n\nfunction curryOuter(a, b) {\n    return function(_) { return a(_, b); };\n}\n\nfunction correct(_, outer) {\n    if (_.type === 'Polygon') {\n        _.coordinates = correctRings(_.coordinates, outer);\n    } else if (_.type === 'MultiPolygon') {\n        _.coordinates = _.coordinates.map(curryOuter(correctRings, outer));\n    }\n    return _;\n}\n\nfunction correctRings(_, outer) {\n    outer = !!outer;\n    _[0] = wind(_[0], !outer);\n    for (var i = 1; i < _.length; i++) {\n        _[i] = wind(_[i], outer);\n    }\n    return _;\n}\n\nfunction wind(_, dir) {\n    return cw(_) === dir ? _ : _.reverse();\n}\n\nfunction cw(_) {\n    return geojsonArea.ring(_) >= 0;\n}\n","var wgs84 = require('wgs84');\n\nmodule.exports.geometry = geometry;\nmodule.exports.ring = ringArea;\n\nfunction geometry(_) {\n    if (_.type === 'Polygon') return polygonArea(_.coordinates);\n    else if (_.type === 'MultiPolygon') {\n        var area = 0;\n        for (var i = 0; i < _.coordinates.length; i++) {\n            area += polygonArea(_.coordinates[i]);\n        }\n        return area;\n    } else {\n        return null;\n    }\n}\n\nfunction polygonArea(coords) {\n    var area = 0;\n    if (coords && coords.length > 0) {\n        area += Math.abs(ringArea(coords[0]));\n        for (var i = 1; i < coords.length; i++) {\n            area -= Math.abs(ringArea(coords[i]));\n        }\n    }\n    return area;\n}\n\n/**\n * Calculate the approximate area of the polygon were it projected onto\n *     the earth.  Note that this area will be positive if ring is oriented\n *     clockwise, otherwise it will be negative.\n *\n * Reference:\n * Robert. G. Chamberlain and William H. Duquette, \"Some Algorithms for\n *     Polygons on a Sphere\", JPL Publication 07-03, Jet Propulsion\n *     Laboratory, Pasadena, CA, June 2007 http://trs-new.jpl.nasa.gov/dspace/handle/2014/40409\n *\n * Returns:\n * {float} The approximate signed geodesic area of the polygon in square\n *     meters.\n */\n\nfunction ringArea(coords) {\n    var area = 0;\n\n    if (coords.length > 2) {\n        var p1, p2;\n        for (var i = 0; i < coords.length - 1; i++) {\n            p1 = coords[i];\n            p2 = coords[i + 1];\n            area += rad(p2[0] - p1[0]) * (2 + Math.sin(rad(p1[1])) + Math.sin(rad(p2[1])));\n        }\n\n        area = area * wgs84.RADIUS * wgs84.RADIUS / 2;\n    }\n\n    return area;\n}\n\nfunction rad(_) {\n    return _ * Math.PI / 180;\n}\n","module.exports.RADIUS = 6378137;\nmodule.exports.FLATTENING = 1/298.257223563;\nmodule.exports.POLAR_RADIUS = 6356752.3142;\n","module.exports={\n  \"$version\": 4,\n  \"$root\": {\n    \"version\": {\n      \"required\": true,\n      \"type\": \"enum\",\n      \"values\": [\n        4\n      ],\n      \"doc\": \"Stylesheet version number. Must be 4.\"\n    },\n    \"constants\": {\n      \"type\": \"constants\",\n      \"doc\": \"An object of constants to be referenced in layers.\"\n    },\n    \"sources\": {\n      \"required\": true,\n      \"type\": \"sources\",\n      \"doc\": \"Data source specifications for layers to pull from.\"\n    },\n    \"layers\": {\n      \"required\": true,\n      \"type\": \"array\",\n      \"value\": \"layer\",\n      \"doc\": \"An array of layers. The order of layers coincides with the order they will be drawn.\"\n    },\n    \"sprite\": {\n      \"type\": \"sprite\",\n      \"doc\": \"Sprite definition.\"\n    },\n    \"glyphs\": {\n      \"type\": \"string\",\n      \"doc\": \"A URL template for loading signed-distance-field glyph sets in PBF format. Valid tokens are {fontstack} and {range}.\"\n    },\n    \"transition\": {\n      \"type\": \"transition\",\n      \"doc\": \"A global transition definition to use as a default across properties.\"\n    }\n  },\n  \"sprite\": [{\n    \"type\": \"string\",\n    \"doc\": \"A base URL for retrieving the sprite image and metadata. The extensions `.png`, `.json` and scale factor `@2x.png` will be automatically appended.\"\n  }],\n  \"constants\": {\n    \"*\": {\n      \"type\": \"*\",\n      \"doc\": \"A constant that will be replaced verbatim in the referencing place. This can be anything, including objects and arrays. All variable names must be prefixed with an `@` symbol.\"\n    }\n  },\n  \"sources\": {\n    \"*\": {\n      \"type\": \"source\"\n    }\n  },\n  \"source\": {\n    \"type\": {\n      \"required\": true,\n      \"type\": \"enum\",\n      \"values\": [\n        \"vector\",\n        \"raster\",\n        \"geojson\",\n        \"video\"\n      ],\n      \"doc\": \"The data type of the source.\"\n    },\n    \"url\": {\n      \"required\": true,\n      \"type\": \"string\",\n      \"doc\": \"A URL, or URL template to retrive the source data.\"\n    },\n    \"tileSize\": {\n      \"type\": \"number\",\n      \"default\": 512,\n      \"doc\": \"The minimum visual size (in px) to display tiles for this layer. Only configurable for raster layers.\"\n    },\n    \"minZoom\": {\n      \"type\": \"number\",\n      \"default\": 0,\n      \"doc\": \"Minimum zoom level for which tiles are available.\"\n    },\n    \"maxZoom\": {\n      \"type\": \"number\",\n      \"default\": 22,\n      \"doc\": \"Maximum zoom level for which tiles are available. Data from tiles at the maxZoom are used when displaying the map at higher zoom levels.\"\n    },\n    \"*\": {\n      \"type\": \"*\",\n      \"doc\": \"Other keys to configure the data source.\"\n    }\n  },\n  \"layer\": {\n    \"id\": {\n      \"type\": \"string\",\n      \"doc\": \"Unique layer name.\"\n    },\n    \"type\": {\n      \"type\": \"enum\",\n      \"values\": [\n        \"fill\",\n        \"line\",\n        \"symbol\",\n        \"raster\",\n        \"background\"\n      ],\n      \"doc\": \"Rendering type of this layer.\"\n    },\n    \"ref\": {\n      \"type\": \"string\",\n      \"doc\": \"References another layer to copy `source`, `source_layer`, `filter`, and `render` properties from. This allows the layers to share processing and be more efficient.\"\n    },\n    \"source\": {\n      \"type\": \"string\",\n      \"doc\": \"Name of a source description to be used for this layer.\"\n    },\n    \"source-layer\": {\n      \"type\": \"string\",\n      \"doc\": \"Layer to use from a vector tile source. Required if the source supports multiple layers.\"\n    },\n    \"min-zoom\": {\n      \"type\": \"number\",\n      \"doc\": \"The minimum zoom level on which the layer gets parsed and appears on.\"\n    },\n    \"max-zoom\": {\n      \"type\": \"number\",\n      \"doc\": \"The maximum zoom level on which the layer gets parsed and appears on.\"\n    },\n    \"interactive\": {\n      \"type\": \"boolean\",\n      \"doc\": \"Enable querying of feature data from this layer for interactivity.\",\n      \"default\": false\n    },\n    \"render\": {\n      \"type\": \"render\",\n      \"doc\": \"Symbolizer type that should be used to visualize this layer. If unspecified or null, this layer is not treated as a symbolizer and only exists to have properties inherited to other layers using ref.\"\n    },\n    \"filter\": {\n      \"type\": \"filter\",\n      \"doc\": \"Array or object of filters or expressions.\"\n    },\n    \"layers\": {\n      \"type\": \"array\",\n      \"value\": \"layer\",\n      \"doc\": \"If `type` is `raster`, the child layers are composited together onto the previous level layer level.\"\n    },\n    \"style\": {\n      \"type\": \"class\",\n      \"doc\": \"Default style properties for this layer.\"\n    },\n    \"style.*\": {\n      \"type\": \"class\",\n      \"doc\": \"Override style properties for this layer. The class name is the part after the first dot.\"\n    }\n  },\n  \"render\": [\n    \"render_fill\",\n    \"render_line\",\n    \"render_symbol\",\n    \"render_raster\",\n    \"render_background\"\n  ],\n  \"render_background\": {\n  },\n  \"render_fill\": {\n  },\n  \"render_line\": {\n    \"line-cap\": {\n      \"type\": \"enum\",\n      \"values\": [\n        \"butt\",\n        \"round\",\n        \"square\"\n      ],\n      \"default\": \"butt\",\n      \"doc\": \"The display of line endings.\"\n    },\n    \"line-join\": {\n      \"type\": \"enum\",\n      \"values\": [\n        \"bevel\",\n        \"round\",\n        \"miter\"\n      ],\n      \"default\": \"miter\",\n      \"doc\": \"The display of lines when joining.\"\n    },\n    \"line-miter-limit\": {\n      \"type\": \"number\",\n      \"default\": 2,\n      \"doc\": \"Used to automatically convert miter joins to bevel joins for sharp angles.\"\n    },\n    \"line-round-limit\": {\n      \"type\": \"number\",\n      \"default\": 1,\n      \"doc\": \"Used to automatically convert round joins to miter joins for shallow angles.\"\n    }\n  },\n  \"render_symbol\": {\n    \"symbol-placement\": {\n        \"type\": \"enum\",\n        \"values\": [\n            \"point\",\n            \"line\"\n        ],\n        \"default\": \"point\",\n        \"doc\": \"Placement of a label relative to its geometry. `Line` can only be used on LineStrings and Polygons.\"\n    },\n    \"symbol-min-distance\": {\n      \"type\": \"number\",\n      \"default\": 250,\n      \"doc\": \"Minimum distance between two symbol anchors (px)\"\n    },\n    \"symbol-avoid-edges\": {\n        \"type\": \"boolean\",\n        \"default\": false,\n        \"doc\": \"If true, the symbols will not cross tile edges. Symbols that cross tile edges may cause collisions in some cases. This property should be set to true if the layer does not have enough padding in the vector tile to prevent collisions, or if it is a point symbol layer placed after a line symbol layer.\"\n    },\n    \"icon-allow-overlap\": {\n        \"type\": \"boolean\",\n        \"default\": false,\n        \"doc\": \"If true, the icon will be visible even if it collides with other icons and text.\"\n    },\n    \"icon-ignore-placement\": {\n        \"type\": \"boolean\",\n        \"default\": false,\n        \"doc\": \"If true, the icon won't affect placement of other icons and text.\"\n    },\n    \"icon-optional\": {\n        \"type\": \"boolean\",\n        \"default\": false,\n        \"doc\": \"If true, text can be shown without its corresponding icon.\"\n    },\n    \"icon-rotation-alignment\": {\n      \"type\": \"enum\",\n      \"values\": [\n        \"map\",\n        \"viewport\"\n      ],\n      \"default\": \"viewport\",\n      \"doc\": \"Orientation of icon when map is rotated\"\n    },\n    \"icon-max-size\": {\n      \"type\": \"number\",\n      \"default\": 1,\n      \"doc\": \"The maximum amount to scale the icon by.\"\n    },\n    \"icon-image\": {\n      \"type\": \"string\",\n      \"doc\": \"A string with {tokens} replaced, referencing the data property to pull from.\"\n    },\n    \"icon-rotate\": {\n      \"type\": \"number\",\n      \"default\": 0,\n      \"doc\": \"Rotates the icon clockwise by the specified number of degrees.\"\n    },\n    \"icon-padding\": {\n      \"type\": \"number\",\n      \"default\": 2,\n      \"doc\": \"Padding value around icon bounding box to avoid icon collisions (px).\"\n    },\n    \"icon-keep-upright\": {\n        \"type\": \"boolean\",\n        \"default\": false,\n        \"doc\": \"If true, the icon may be flipped to prevent it from being rendered upside-down\"\n    },\n    \"icon-offset\": {\n      \"type\": \"array\",\n      \"value\": \"number\",\n      \"length\": 2,\n      \"default\": [\n        0,\n        0\n      ],\n      \"doc\": \"Icon's offset, in pixels. Values are [x, y] where negatives indicate left and up, respectively.\"\n    },\n    \"text-rotation-alignment\": {\n      \"type\": \"enum\",\n      \"values\": [\n        \"map\",\n        \"viewport\"\n      ],\n      \"default\": \"viewport\",\n      \"doc\": \"Orientation of icon or text when map is rotated\"\n    },\n    \"text-field\": {\n      \"type\": \"string\",\n      \"default\": \"\",\n      \"doc\": \"Value to use for a text label. Feature properties are specified using tokens like {field_name}.\"\n    },\n    \"text-font\": {\n      \"type\": \"string\",\n      \"doc\": \"Font stack to use for displaying text.\"\n    },\n    \"text-max-size\": {\n      \"type\": \"number\",\n      \"default\": 16,\n      \"doc\": \"The maximum size text will be displayed.\"\n    },\n    \"text-max-width\": {\n      \"type\": \"number\",\n      \"default\": 15,\n      \"doc\": \"The maximum line width for text wrapping (em).\"\n    },\n    \"text-line-height\": {\n      \"type\": \"number\",\n      \"default\": 1.2,\n      \"doc\": \"Text leading value for multi-line text.\"\n    },\n    \"text-letter-spacing\": {\n      \"type\": \"number\",\n      \"default\": 0,\n      \"doc\": \"Text kerning value (em).\"\n    },\n    \"text-justify\": {\n      \"type\": \"enum\",\n      \"values\": [\n        \"center\",\n        \"left\",\n        \"right\"\n      ],\n      \"default\": \"center\",\n      \"doc\": \"Text justification options.\"\n    },\n    \"text-horizontal-align\": {\n      \"type\": \"enum\",\n      \"values\": [\n        \"left\",\n        \"center\",\n        \"right\"\n      ],\n      \"default\": \"center\",\n      \"doc\": \"Horizontal alignment of the text relative to the anchor.\"\n    },\n    \"text-vertical-align\": {\n      \"type\": \"enum\",\n      \"values\": [\n        \"top\",\n        \"center\",\n        \"bottom\"\n      ],\n      \"default\": \"center\",\n      \"doc\": \"Vertical alignment of the text relative to the anchor.\"\n    },\n    \"text-max-angle\": {\n      \"type\": \"number\",\n      \"default\": 45,\n      \"doc\": \"The maximum angle change, in degrees, allowed between adjacent characters.\"\n    },\n    \"text-rotate\": {\n      \"type\": \"number\",\n      \"default\": 0,\n      \"doc\": \"Rotates the text clockwise by the specified number of degrees.\"\n    },\n    \"text-padding\": {\n      \"type\": \"number\",\n      \"default\": 2,\n      \"doc\": \"Padding value around text bounding box to avoid label collisions (px).\"\n    },\n    \"text-keep-upright\": {\n        \"type\": \"boolean\",\n        \"default\": true,\n        \"doc\": \"If true, the direction of the text may be flipped to prevent it from being rendered upside-down\"\n    },\n    \"text-transform\": {\n      \"type\": \"enum\",\n      \"values\": [\n        \"none\",\n        \"uppercase\",\n        \"lowercase\"\n      ],\n      \"default\": \"none\",\n      \"doc\": \"Specifies how to capitalize text, similar to the CSS `text-transform` property.\"\n    },\n    \"text-offset\": {\n      \"type\": \"array\",\n      \"value\": \"number\",\n      \"length\": 2,\n      \"default\": [\n        0,\n        0\n      ]\n    },\n    \"text-allow-overlap\": {\n      \"type\": \"boolean\",\n      \"default\": false,\n      \"doc\": \"If true, the text will be visible even if it collides with other icons and labels.\"\n    },\n    \"text-ignore-placement\": {\n      \"type\": \"boolean\",\n      \"default\": false,\n      \"doc\": \"If true, the text won't affect placement of other icons and labels.\"\n    },\n    \"text-optional\": {\n        \"type\": \"boolean\",\n        \"default\": false,\n        \"doc\": \"If true, icons can be shown without their corresponding text.\"\n    }\n  },\n  \"render_raster\": {\n    \"raster-size\": {\n      \"type\": \"number\",\n      \"function\": true,\n      \"default\": 256,\n      \"doc\": \"The texture image size (in pixels) vector layers will be rasterized at. Will automatically by scaled to match the visual tile size.\"\n    },\n    \"raster-blur\": {\n      \"type\": \"number\",\n      \"function\": true,\n      \"default\": 0,\n      \"doc\": \"Blur radius to apply to the raster texture before display.\"\n    }\n  },\n  \"filter\": [\n    {\n      \"type\": \"filter_expression\",\n      \"doc\": \"Various filter expressions. Unless overridden by parent, these are interpreted as `AND`.\"\n    },\n    {\n      \"type\": \"array\",\n      \"value\": \"filter_expression\",\n      \"doc\": \"Various filter expressions. Unless overridden by parent, these are interpreted as `OR`.\"\n    }\n  ],\n  \"filter_expression\": {\n    \"&\": {\n      \"type\": \"filter\",\n      \"doc\": \"AND operator.\"\n    },\n    \"|\": {\n      \"type\": \"filter\",\n      \"doc\": \"OR operator.\"\n    },\n    \"^\": {\n      \"type\": \"filter\",\n      \"doc\": \"XOR operator.\"\n    },\n    \"!\": {\n      \"type\": \"filter\",\n      \"doc\": \"NOR operator.\"\n    },\n    \"$type\": {\n      \"type\": \"enum\",\n      \"values\": [\n        \"Point\",\n        \"LineString\",\n        \"Polygon\"\n      ],\n      \"doc\": \"Geometry type that features must match.\"\n    },\n    \"*\": [\n      {\n        \"type\": \"filter_comparison\",\n        \"doc\": \"Arbitarily named feature member. A comparison object defining a filter expression.\"\n      },\n      {\n        \"type\": \"filter_value\",\n        \"doc\": \"Arbitarily named feature member. A filter_value implies the equality (string/number/boolean) or set membership operator (array).\"\n      }\n    ]\n  },\n  \"filter_comparison\": {\n    \"==\": {\n      \"type\": \"filter_value\",\n      \"doc\": \"Equality operator.\"\n    },\n    \"!=\": {\n      \"type\": \"filter_value\",\n      \"doc\": \"Inequality operator.\"\n    },\n    \">\": {\n      \"type\": \"filter_value\",\n      \"doc\": \"Greater than operator.\"\n    },\n    \">=\": {\n      \"type\": \"filter_value\",\n      \"doc\": \"Greater or equal than operator.\"\n    },\n    \"<\": {\n      \"type\": \"filter_value\",\n      \"doc\": \"Less than operator.\"\n    },\n    \"<=\": {\n      \"type\": \"filter_value\",\n      \"doc\": \"Less than or equal operator.\"\n    },\n    \"in\": {\n      \"type\": \"array\",\n      \"value\": \"filter_primitive\",\n      \"doc\": \"Set member operator.\"\n    },\n    \"!in\": {\n      \"type\": \"array\",\n      \"value\": \"filter_primitive\",\n      \"doc\": \"Not in set operator.\"\n    }\n  },\n  \"filter_value\": [\n    {\n      \"type\": \"filter_primitive\"\n    },\n    {\n      \"type\": \"array\",\n      \"value\": \"filter_primitive\"\n    }\n  ],\n  \"filter_primitive\": [\n    {\n      \"type\": \"string\"\n    },\n    {\n      \"type\": \"number\"\n    },\n    {\n      \"type\": \"boolean\"\n    }\n  ],\n  \"function\": {\n    \"stops\": {\n      \"type\": \"array\",\n      \"required\": true,\n      \"doc\": \"An array of stops.\",\n      \"value\": \"function_stop\"\n    },\n    \"base\": {\n      \"type\": \"number\",\n      \"default\": 1,\n      \"doc\": \"The exponential base of the interpolation curve. It controls the rate at which the result increases. Higher values make the result increase more towards the high end of the range. With `1` the stops are interpolated linearly.\"\n    }\n  },\n  \"function_stop\": {\n    \"type\": \"array\",\n    \"value\": [\"number\", \"color\"],\n    \"length\": 2,\n    \"doc\": \"Zoom level and value pair.\"\n  },\n  \"class\": [\n    \"class_fill\",\n    \"class_line\",\n    \"class_symbol\",\n    \"class_raster\",\n    \"class_background\"\n  ],\n  \"class_fill\": {\n    \"fill-antialias\": {\n      \"type\": \"boolean\",\n      \"default\": true,\n      \"function\": true,\n      \"doc\": \"Whether or not the fill should be antialiased.\"\n    },\n    \"fill-opacity\": {\n      \"type\": \"number\",\n      \"function\": true,\n      \"default\": 1,\n      \"transition\": true\n    },\n    \"fill-color\": {\n      \"type\": \"color\",\n      \"default\": [\n        0,\n        0,\n        0,\n        1\n      ],\n      \"function\": true,\n      \"transition\": true\n    },\n    \"fill-outline-color\": {\n      \"type\": \"color\",\n      \"doc\": \"The outline color of the fill. Matches the value of `fill-color` if unspecified.\",\n      \"function\": true,\n      \"transition\": true\n    },\n    \"fill-translate\": {\n      \"type\": \"array\",\n      \"value\": \"number\",\n      \"length\": 2,\n      \"default\": [\n        0,\n        0\n      ],\n      \"function\": true,\n      \"transition\": true,\n      \"doc\": \"The geometry's offset, in pixels. Values are [x, y] where negatives indicate left and up, respectively.\"\n    },\n    \"fill-translate-anchor\": {\n      \"type\": \"enum\",\n      \"values\": [\n        \"map\",\n        \"viewport\"\n      ],\n      \"doc\": \"Control whether the translation is relative to the map (north) or viewport (screen)\",\n      \"default\": \"map\"\n    },\n    \"fill-image\": {\n      \"type\": \"string\",\n      \"doc\": \"Name of image in sprite to use for drawing image fills.\"\n    }\n  },\n  \"class_line\": {\n    \"line-opacity\": {\n      \"type\": \"number\",\n      \"function\": true,\n      \"default\": 1,\n      \"transition\": true\n    },\n    \"line-color\": {\n      \"type\": \"color\",\n      \"default\": [\n        0,\n        0,\n        0,\n        1\n      ],\n      \"function\": true,\n      \"transition\": true\n    },\n    \"line-translate\": {\n      \"type\": \"array\",\n      \"value\": \"number\",\n      \"length\": 2,\n      \"default\": [\n        0,\n        0\n      ],\n      \"function\": true,\n      \"transition\": true,\n      \"doc\": \"The geometry's offset, in pixels. Values are [x, y] where negatives indicate left and up, respectively.\"\n    },\n    \"line-translate-anchor\": {\n      \"type\": \"enum\",\n      \"values\": [\n        \"map\",\n        \"viewport\"\n      ],\n      \"doc\": \"Control whether the translation is relative to the map (north) or viewport (screen)\",\n      \"default\": \"map\"\n    },\n    \"line-width\": {\n      \"type\": \"number\",\n      \"default\": 1,\n      \"function\": true,\n      \"transition\": true,\n      \"doc\": \"Line width (in px)\"\n    },\n    \"line-offset\": {\n      \"type\": \"number\",\n      \"default\": 0,\n      \"doc\": \"Line casing where `line-offset` indicates total width. @TODO rename?\",\n      \"function\": true,\n      \"transition\": true\n    },\n    \"line-blur\": {\n      \"type\": \"number\",\n      \"default\": 0,\n      \"function\": true,\n      \"transition\": true,\n      \"doc\": \"Line blur, in pixels.\"\n    },\n    \"line-dasharray\": {\n      \"type\": \"array\",\n      \"value\": \"number\",\n      \"length\": 2,\n      \"default\": [\n        1,\n        -1\n      ],\n      \"function\": true,\n      \"transition\": true\n    },\n    \"line-image\": {\n      \"type\": \"string\",\n      \"doc\": \"Name of image in sprite to use for drawing image lines.\"\n    }\n  },\n  \"class_symbol\": {\n    \"icon-opacity\": {\n      \"type\": \"number\",\n      \"default\": 1,\n      \"function\": true,\n      \"transition\": true\n    },\n    \"icon-size\": {\n      \"type\": \"number\",\n      \"default\": 1,\n      \"function\": true,\n      \"transition\": true,\n      \"doc\": \"The amount to scale the icon by. 1 is original size, 3 triples the size.\"\n    },\n    \"icon-color\": {\n      \"type\": \"color\",\n      \"default\": [\n        0,\n        0,\n        0,\n        1\n      ],\n      \"function\": true,\n      \"transition\": true,\n      \"doc\": \"The color of the icon. This can only be used with sdf icons.\"\n    },\n    \"icon-halo-color\": {\n      \"type\": \"color\",\n      \"default\": [\n        0,\n        0,\n        0,\n        0\n      ],\n      \"function\": true,\n      \"transition\": true,\n      \"doc\": \"The color of the icon's halo. Icon halos can only be used with sdf icons.\"\n    },\n    \"icon-halo-width\": {\n      \"type\": \"number\",\n      \"default\": 0,\n      \"function\": true,\n      \"transition\": true,\n      \"doc\": \"How far away the halo is from the icon outline, in pixels.\"\n    },\n    \"icon-halo-blur\": {\n      \"type\": \"number\",\n      \"default\": 0,\n      \"function\": true,\n      \"transition\": true,\n      \"doc\": \"Fade out the halo towards the outside, in pixels.\"\n    },\n    \"icon-translate\": {\n      \"type\": \"array\",\n      \"value\": \"number\",\n      \"length\": 2,\n      \"default\": [\n        0,\n        0\n      ],\n      \"function\": true,\n      \"transition\": true,\n      \"doc\": \"An icon's offset, in pixels. Values are [x, y] where negatives indicate left and up, respectively.\"\n    },\n    \"icon-translate-anchor\": {\n      \"type\": \"enum\",\n      \"values\": [\n        \"map\",\n        \"viewport\"\n      ],\n      \"doc\": \"Control whether the translation is relative to the map (north) or viewport (screen)\",\n      \"default\": \"map\"\n    },\n    \"text-opacity\": {\n      \"type\": \"number\",\n      \"default\": 1,\n      \"function\": true,\n      \"transition\": true\n    },\n    \"text-size\": {\n      \"type\": \"number\",\n      \"default\": 16,\n      \"function\": true,\n      \"transition\": true,\n      \"doc\": \"Font size in pixels. If unspecified, the text will be as big as allowed by the layer definition.\"\n    },\n    \"text-color\": {\n      \"type\": \"color\",\n      \"default\": [\n        0,\n        0,\n        0,\n        1\n      ],\n      \"function\": true,\n      \"transition\": true\n    },\n    \"text-halo-color\": {\n      \"type\": \"color\",\n      \"default\": [\n        0,\n        0,\n        0,\n        0\n      ],\n      \"function\": true,\n      \"transition\": true\n    },\n    \"text-halo-width\": {\n      \"type\": \"number\",\n      \"default\": 0,\n      \"function\": true,\n      \"transition\": true,\n      \"doc\": \"How far away the halo is from the font outline, in pixels. Max text halo width is 1/4 of the font-size (px).\"\n    },\n    \"text-halo-blur\": {\n      \"type\": \"number\",\n      \"default\": 0,\n      \"function\": true,\n      \"transition\": true,\n      \"doc\": \"Fade out the halo towards the outside, in pixels.\"\n    },\n    \"text-translate\": {\n      \"type\": \"array\",\n      \"value\": \"number\",\n      \"length\": 2,\n      \"default\": [\n        0,\n        0\n      ],\n      \"function\": true,\n      \"transition\": true,\n      \"doc\": \"A label's offset, in pixels. Values are [x, y] where negatives indicate left and up, respectively.\"\n    },\n    \"text-translate-anchor\": {\n      \"type\": \"enum\",\n      \"values\": [\n        \"map\",\n        \"viewport\"\n      ],\n      \"doc\": \"Control whether the translation is relative to the map (north) or viewport (screen)\",\n      \"default\": \"map\"\n    }\n  },\n  \"class_raster\": {\n    \"raster-opacity\": {\n      \"type\": \"number\",\n      \"default\": 1,\n      \"transition\": true\n    },\n    \"raster-hue-rotate\": {\n      \"type\": \"number\",\n      \"default\": 0,\n      \"function\": true,\n      \"transition\": true,\n      \"doc\": \"Rotates hues around the color wheel by the specified number of degrees.\"\n    },\n    \"raster-brightness\": {\n      \"type\": \"array\",\n      \"value\": \"number\",\n      \"length\": 2,\n      \"default\": [\n        0,\n        1\n      ],\n      \"function\": true,\n      \"transition\": true\n    },\n    \"raster-saturation\": {\n      \"type\": \"number\",\n      \"default\": 0,\n      \"function\": true,\n      \"transition\": true\n    },\n    \"raster-contrast\": {\n      \"type\": \"number\",\n      \"default\": 0,\n      \"function\": true,\n      \"transition\": true\n    },\n    \"raster-fade\": {\n      \"type\": \"number\",\n      \"default\": 0,\n      \"function\": true,\n      \"transition\": true,\n      \"doc\": \"Duration of the fade when a new tile is added. @TODO rename?\"\n    }\n  },\n  \"class_background\": {\n    \"background-color\": {\n      \"type\": \"color\",\n      \"default\": [\n        0,\n        0,\n        0,\n        1\n      ],\n      \"function\": true,\n      \"transition\": true\n    },\n    \"background-image\": {\n      \"type\": \"string\"\n    }\n  },\n  \"transition\": {\n    \"duration\": {\n      \"type\": \"number\",\n      \"default\": 300,\n      \"doc\": \"Time in milliseconds that it takes for transitions to complete.\"\n    },\n    \"delay\": {\n      \"type\": \"number\",\n      \"default\": 0,\n      \"doc\": \"Time in milliseconds before a transition begins.\"\n    }\n  }\n}\n","(function (Buffer){\n'use strict';\n\nvar ieee754 = require('ieee754');\n\nmodule.exports = Protobuf;\nfunction Protobuf(buf) {\n    this.buf = buf;\n    this.pos = 0;\n}\n\nProtobuf.prototype = {\n    get length() { return this.buf.length; }\n};\n\nProtobuf.Varint = 0;\nProtobuf.Int64 = 1;\nProtobuf.Message = 2;\nProtobuf.String = 2;\nProtobuf.Packed = 2;\nProtobuf.Int32 = 5;\n\nProtobuf.prototype.destroy = function() {\n    this.buf = null;\n};\n\n// === READING =================================================================\n\nProtobuf.prototype.readUInt32 = function() {\n    var val = this.buf.readUInt32LE(this.pos);\n    this.pos += 4;\n    return val;\n};\n\nProtobuf.prototype.readUInt64 = function() {\n    var val = this.buf.readUInt64LE(this.pos);\n    this.pos += 8;\n    return val;\n};\n\nProtobuf.prototype.readDouble = function() {\n    var val = ieee754.read(this.buf, this.pos, true, 52, 8);\n    this.pos += 8;\n    return val;\n};\n\nProtobuf.prototype.readVarint = function() {\n    // TODO: bounds checking\n    var pos = this.pos;\n    if (this.buf[pos] <= 0x7f) {\n        this.pos++;\n        return this.buf[pos];\n    } else if (this.buf[pos + 1] <= 0x7f) {\n        this.pos += 2;\n        return (this.buf[pos] & 0x7f) | (this.buf[pos + 1] << 7);\n    } else if (this.buf[pos + 2] <= 0x7f) {\n        this.pos += 3;\n        return (this.buf[pos] & 0x7f) | (this.buf[pos + 1] & 0x7f) << 7 | (this.buf[pos + 2]) << 14;\n    } else if (this.buf[pos + 3] <= 0x7f) {\n        this.pos += 4;\n        return (this.buf[pos] & 0x7f) | (this.buf[pos + 1] & 0x7f) << 7 | (this.buf[pos + 2] & 0x7f) << 14 | (this.buf[pos + 3]) << 21;\n    } else if (this.buf[pos + 4] <= 0x7f) {\n        this.pos += 5;\n        return ((this.buf[pos] & 0x7f) | (this.buf[pos + 1] & 0x7f) << 7 | (this.buf[pos + 2] & 0x7f) << 14 | (this.buf[pos + 3]) << 21) + (this.buf[pos + 4] * 268435456);\n    } else {\n        this.skip(Protobuf.Varint);\n        return 0;\n        // throw new Error(\"TODO: Handle 6+ byte varints\");\n    }\n};\n\nProtobuf.prototype.readSVarint = function() {\n    var num = this.readVarint();\n    if (num > 2147483647) throw new Error('TODO: Handle numbers >= 2^30');\n    // zigzag encoding\n    return ((num >> 1) ^ -(num & 1));\n};\n\nProtobuf.prototype.readString = function() {\n    var bytes = this.readVarint();\n    // TODO: bounds checking\n    var chr = String.fromCharCode;\n    var b = this.buf;\n    var p = this.pos;\n    var end = this.pos + bytes;\n    var str = '';\n    while (p < end) {\n        if (b[p] <= 0x7F) str += chr(b[p++]);\n        else if (b[p] <= 0xBF) throw new Error('Invalid UTF-8 codepoint: ' + b[p]);\n        else if (b[p] <= 0xDF) str += chr((b[p++] & 0x1F) << 6 | (b[p++] & 0x3F));\n        else if (b[p] <= 0xEF) str += chr((b[p++] & 0x1F) << 12 | (b[p++] & 0x3F) << 6 | (b[p++] & 0x3F));\n        else if (b[p] <= 0xF7) p += 4; // We can't handle these codepoints in JS, so skip.\n        else if (b[p] <= 0xFB) p += 5;\n        else if (b[p] <= 0xFD) p += 6;\n        else throw new Error('Invalid UTF-8 codepoint: ' + b[p]);\n    }\n    this.pos += bytes;\n    return str;\n};\n\nProtobuf.prototype.readBuffer = function() {\n    var bytes = this.readVarint();\n    var buffer = this.buf.subarray(this.pos, this.pos + bytes);\n    this.pos += bytes;\n    return buffer;\n};\n\nProtobuf.prototype.readPacked = function(type) {\n    // TODO: bounds checking\n    var bytes = this.readVarint();\n    var end = this.pos + bytes;\n    var array = [];\n    while (this.pos < end) {\n        array.push(this['read' + type]());\n    }\n    return array;\n};\n\nProtobuf.prototype.skip = function(val) {\n    // TODO: bounds checking\n    var type = val & 0x7;\n    switch (type) {\n        /* varint */ case Protobuf.Varint: while (this.buf[this.pos++] > 0x7f); break;\n        /* 64 bit */ case Protobuf.Int64: this.pos += 8; break;\n        /* length */ case Protobuf.Message: var bytes = this.readVarint(); this.pos += bytes; break;\n        /* 32 bit */ case Protobuf.Int32: this.pos += 4; break;\n        default: throw new Error('Unimplemented type: ' + type);\n    }\n};\n\n// === WRITING =================================================================\n\nProtobuf.prototype.writeTag = function(tag, type) {\n    this.writeVarint((tag << 3) | type);\n};\n\nProtobuf.prototype.realloc = function(min) {\n    var length = this.buf.length;\n    while (length < this.pos + min) length *= 2;\n    if (length != this.buf.length) {\n        var buf = new Buffer(length);\n        this.buf.copy(buf);\n        this.buf = buf;\n    }\n};\n\nProtobuf.prototype.finish = function() {\n    return this.buf.slice(0, this.pos);\n};\n\nProtobuf.prototype.writePacked = function(type, tag, items) {\n    if (!items.length) return;\n\n    var message = new Protobuf();\n    for (var i = 0; i < items.length; i++) {\n        message['write' + type](items[i]);\n    }\n    var data = message.finish();\n\n    this.writeTag(tag, Protobuf.Packed);\n    this.writeBuffer(data);\n};\n\nProtobuf.prototype.writeUInt32 = function(val) {\n    this.realloc(4);\n    this.buf.writeUInt32LE(val, this.pos);\n    this.pos += 4;\n};\n\nProtobuf.prototype.writeTaggedUInt32 = function(tag, val) {\n    this.writeTag(tag, Protobuf.Int32);\n    this.writeUInt32(val);\n};\n\nProtobuf.prototype.writeVarint = function(val) {\n    val = Number(val);\n    if (isNaN(val)) {\n        val = 0;\n    }\n\n    if (val <= 0x7f) {\n        this.realloc(1);\n        this.buf[this.pos++] = val;\n    } else if (val <= 0x3fff) {\n        this.realloc(2);\n        this.buf[this.pos++] = 0x80 | ((val >>> 0) & 0x7f);\n        this.buf[this.pos++] = 0x00 | ((val >>> 7) & 0x7f);\n    } else if (val <= 0x1ffffff) {\n        this.realloc(3);\n        this.buf[this.pos++] = 0x80 | ((val >>> 0) & 0x7f);\n        this.buf[this.pos++] = 0x80 | ((val >>> 7) & 0x7f);\n        this.buf[this.pos++] = 0x00 | ((val >>> 14) & 0x7f);\n    } else if (val <= 0xfffffff) {\n        this.realloc(4);\n        this.buf[this.pos++] = 0x80 | ((val >>> 0) & 0x7f);\n        this.buf[this.pos++] = 0x80 | ((val >>> 7) & 0x7f);\n        this.buf[this.pos++] = 0x80 | ((val >>> 14) & 0x7f);\n        this.buf[this.pos++] = 0x00 | ((val >>> 21) & 0x7f);\n    } else {\n        while (val > 0) {\n            var b = val & 0x7f;\n            val = Math.floor(val / 128);\n            if (val > 0) b |= 0x80\n            this.realloc(1);\n            this.buf[this.pos++] = b;\n        }\n    }\n};\n\nProtobuf.prototype.writeTaggedVarint = function(tag, val) {\n    this.writeTag(tag, Protobuf.Varint);\n    this.writeVarint(val);\n};\n\nProtobuf.prototype.writeSVarint = function(val) {\n    if (val >= 0) {\n        this.writeVarint(val * 2);\n    } else {\n        this.writeVarint(val * -2 - 1);\n    }\n};\n\nProtobuf.prototype.writeTaggedSVarint = function(tag, val) {\n    this.writeTag(tag, Protobuf.Varint);\n    this.writeSVarint(val);\n};\n\nProtobuf.prototype.writeBoolean = function(val) {\n    this.writeVarint(Boolean(val));\n};\n\nProtobuf.prototype.writeTaggedBoolean = function(tag, val) {\n    this.writeTaggedVarint(tag, Boolean(val));\n};\n\nProtobuf.prototype.writeString = function(str) {\n    str = String(str);\n    var bytes = Buffer.byteLength(str);\n    this.writeVarint(bytes);\n    this.realloc(bytes);\n    this.buf.write(str, this.pos);\n    this.pos += bytes;\n};\n\nProtobuf.prototype.writeTaggedString = function(tag, str) {\n    this.writeTag(tag, Protobuf.String);\n    this.writeString(str);\n};\n\nProtobuf.prototype.writeFloat = function(val) {\n    this.realloc(4);\n    this.buf.writeFloatLE(val, this.pos);\n    this.pos += 4;\n};\n\nProtobuf.prototype.writeTaggedFloat = function(tag, val) {\n    this.writeTag(tag, Protobuf.Int32);\n    this.writeFloat(val);\n};\n\nProtobuf.prototype.writeDouble = function(val) {\n    this.realloc(8);\n    this.buf.writeDoubleLE(val, this.pos);\n    this.pos += 8;\n};\n\nProtobuf.prototype.writeTaggedDouble = function(tag, val) {\n    this.writeTag(tag, Protobuf.Int64);\n    this.writeDouble(val);\n};\n\nProtobuf.prototype.writeBuffer = function(buffer) {\n    var bytes = buffer.length;\n    this.writeVarint(bytes);\n    this.realloc(bytes);\n    buffer.copy(this.buf, this.pos);\n    this.pos += bytes;\n};\n\nProtobuf.prototype.writeTaggedBuffer = function(tag, buffer) {\n    this.writeTag(tag, Protobuf.String);\n    this.writeBuffer(buffer);\n};\n\nProtobuf.prototype.writeMessage = function(tag, protobuf) {\n    var buffer = protobuf.finish();\n    this.writeTag(tag, Protobuf.Message);\n    this.writeBuffer(buffer);\n};\n\n}).call(this,require(\"buffer\").Buffer)","'use strict';\n\nmodule.exports = Point;\n\nfunction Point(x, y) {\n    this.x = x;\n    this.y = y;\n}\n\nPoint.prototype = {\n    clone: function() { return new Point(this.x, this.y); },\n\n    add:     function(p) { return this.clone()._add(p);     },\n    sub:     function(p) { return this.clone()._sub(p);     },\n    mult:    function(k) { return this.clone()._mult(k);    },\n    div:     function(k) { return this.clone()._div(k);     },\n    rotate:  function(a) { return this.clone()._rotate(a);  },\n    matMult: function(m) { return this.clone()._matMult(m); },\n    unit:    function() { return this.clone()._unit(); },\n    perp:    function() { return this.clone()._perp(); },\n    round:   function() { return this.clone()._round(); },\n\n    mag: function() {\n        return Math.sqrt(this.x * this.x + this.y * this.y);\n    },\n\n    equals: function(p) {\n        return this.x === p.x &&\n               this.y === p.y;\n    },\n\n    dist: function(p) {\n        return Math.sqrt(this.distSqr(p));\n    },\n\n    distSqr: function(p) {\n        var dx = p.x - this.x,\n            dy = p.y - this.y;\n        return dx * dx + dy * dy;\n    },\n\n    angle: function() {\n        return Math.atan2(this.y, this.x);\n    },\n\n    angleTo: function(b) {\n        return Math.atan2(this.y - b.y, this.x - b.x);\n    },\n\n    angleWith: function(b) {\n        return this.angleWithSep(b.x, b.y);\n    },\n\n    // Find the angle of the two vectors, solving the formula for the cross product a x b = |a||b|sin(θ) for θ.\n    angleWithSep: function(x, y) {\n        return Math.atan2(\n            this.x * y - this.y * x,\n            this.x * x + this.y * y);\n    },\n\n    _matMult: function(m) {\n        var x = m[0] * this.x + m[1] * this.y,\n            y = m[2] * this.x + m[3] * this.y;\n        this.x = x;\n        this.y = y;\n        return this;\n    },\n\n    _add: function(p) {\n        this.x += p.x;\n        this.y += p.y;\n        return this;\n    },\n\n    _sub: function(p) {\n        this.x -= p.x;\n        this.y -= p.y;\n        return this;\n    },\n\n    _mult: function(k) {\n        this.x *= k;\n        this.y *= k;\n        return this;\n    },\n\n    _div: function(k) {\n        this.x /= k;\n        this.y /= k;\n        return this;\n    },\n\n    _unit: function() {\n        this._div(this.mag());\n        return this;\n    },\n\n    _perp: function() {\n        var y = this.y;\n        this.y = this.x;\n        this.x = -y;\n        return this;\n    },\n\n    _rotate: function(angle) {\n        var cos = Math.cos(angle),\n            sin = Math.sin(angle),\n            x = cos * this.x - sin * this.y,\n            y = sin * this.x + cos * this.y;\n        this.x = x;\n        this.y = y;\n        return this;\n    },\n\n    _round: function() {\n        this.x = Math.round(this.x);\n        this.y = Math.round(this.y);\n        return this;\n    }\n};\n\n// constructs Point from an array if necessary\nPoint.convert = function (a) {\n    if (a instanceof Point) {\n        return a;\n    }\n    if (Array.isArray(a)) {\n        return new Point(a[0], a[1]);\n    }\n    return a;\n};\n","(function() {\n  var slice = [].slice;\n\n  function queue(parallelism) {\n    var q,\n        tasks = [],\n        started = 0, // number of tasks that have been started (and perhaps finished)\n        active = 0, // number of tasks currently being executed (started but not finished)\n        remaining = 0, // number of tasks not yet finished\n        popping, // inside a synchronous task callback?\n        error = null,\n        await = noop,\n        all;\n\n    if (!parallelism) parallelism = Infinity;\n\n    function pop() {\n      while (popping = started < tasks.length && active < parallelism) {\n        var i = started++,\n            t = tasks[i],\n            a = slice.call(t, 1);\n        a.push(callback(i));\n        ++active;\n        t[0].apply(null, a);\n      }\n    }\n\n    function callback(i) {\n      return function(e, r) {\n        --active;\n        if (error != null) return;\n        if (e != null) {\n          error = e; // ignore new tasks and squelch active callbacks\n          started = remaining = NaN; // stop queued tasks from starting\n          notify();\n        } else {\n          tasks[i] = r;\n          if (--remaining) popping || pop();\n          else notify();\n        }\n      };\n    }\n\n    function notify() {\n      if (error != null) await(error);\n      else if (all) await(error, tasks);\n      else await.apply(null, [error].concat(tasks));\n    }\n\n    return q = {\n      defer: function() {\n        if (!error) {\n          tasks.push(arguments);\n          ++remaining;\n          pop();\n        }\n        return q;\n      },\n      await: function(f) {\n        await = f;\n        all = false;\n        if (!remaining) notify();\n        return q;\n      },\n      awaitAll: function(f) {\n        await = f;\n        all = true;\n        if (!remaining) notify();\n        return q;\n      }\n    };\n  }\n\n  function noop() {}\n\n  queue.version = \"1.0.7\";\n  if (typeof define === \"function\" && define.amd) define(function() { return queue; });\n  else if (typeof module === \"object\" && module.exports) module.exports = queue;\n  else this.queue = queue;\n})();\n","/*\n (c) 2013, Vladimir Agafonkin\n RBush, a JavaScript library for high-performance 2D spatial indexing of points and rectangles.\n https://github.com/mourner/rbush\n*/\n\n(function () { 'use strict';\n\nfunction rbush(maxEntries, format) {\n\n    // jshint newcap: false, validthis: true\n    if (!(this instanceof rbush)) { return new rbush(maxEntries, format); }\n\n    // max entries in a node is 9 by default; min node fill is 40% for best performance\n    this._maxEntries = Math.max(4, maxEntries || 9);\n    this._minEntries = Math.max(2, Math.ceil(this._maxEntries * 0.4));\n\n    if (format) {\n        this._initFormat(format);\n    }\n\n    this.clear();\n}\n\nrbush.prototype = {\n\n    all: function () {\n        return this._all(this.data, []);\n    },\n\n    search: function (bbox) {\n\n        var node = this.data,\n            result = [];\n\n        if (!this._intersects(bbox, node.bbox)) { return result; }\n\n        var nodesToSearch = [],\n            i, len, child, childBBox;\n\n        while (node) {\n            for (i = 0, len = node.children.length; i < len; i++) {\n                child = node.children[i];\n                childBBox = node.leaf ? this.toBBox(child) : child.bbox;\n\n                if (this._intersects(bbox, childBBox)) {\n\n                    if (node.leaf) {\n                        result.push(child);\n\n                    } else if (this._contains(bbox, childBBox)) {\n                        this._all(child, result);\n\n                    } else {\n                        nodesToSearch.push(child);\n                    }\n                }\n            }\n\n            node = nodesToSearch.pop();\n        }\n\n        return result;\n    },\n\n    load: function (data) {\n        if (!(data && data.length)) { return this; }\n\n        if (data.length < this._minEntries) {\n            for (var i = 0, len = data.length; i < len; i++) {\n                this.insert(data[i]);\n            }\n            return this;\n        }\n\n        // recursively build the tree with the given data from stratch using OMT algorithm\n        var node = this._build(data.slice(), 0);\n\n        if (!this.data.children.length) {\n            // save as is if tree is empty\n            this.data = node;\n\n        } else if (this.data.height === node.height) {\n            // split root if trees have the same height\n            this._splitRoot(this.data, node);\n\n        } else {\n            if (this.data.height < node.height) {\n                // swap trees if inserted one is bigger\n                var tmpNode = this.data;\n                this.data = node;\n                node = tmpNode;\n            }\n\n            // insert the small tree into the large tree at appropriate level\n            this._insert(node, this.data.height - node.height - 1, true);\n        }\n\n        return this;\n    },\n\n    insert: function (item) {\n        if (item) {\n            this._insert(item, this.data.height - 1);\n        }\n        return this;\n    },\n\n    clear: function () {\n        this.data = {\n            children: [],\n            leaf: true,\n            bbox: this._empty(),\n            height: 1\n        };\n        return this;\n    },\n\n    remove: function (item) {\n        if (!item) { return this; }\n\n        var node = this.data,\n            bbox = this.toBBox(item),\n            path = [],\n            indexes = [],\n            i, parent, index, goingUp;\n\n        // depth-first iterative tree traversal\n        while (node || path.length) {\n\n            if (!node) { // go up\n                node = path.pop();\n                parent = path[path.length - 1];\n                i = indexes.pop();\n                goingUp = true;\n            }\n\n            if (node.leaf) { // check current node\n                index = node.children.indexOf(item);\n\n                if (index !== -1) {\n                    // item found, remove the item and condense tree upwards\n                    node.children.splice(index, 1);\n                    path.push(node);\n                    this._condense(path);\n                    return this;\n                }\n            }\n\n            if (!goingUp && !node.leaf && this._contains(node.bbox, bbox)) { // go down\n                path.push(node);\n                indexes.push(i);\n                i = 0;\n                parent = node;\n                node = node.children[0];\n\n            } else if (parent) { // go right\n                i++;\n                node = parent.children[i];\n                goingUp = false;\n\n            } else { // nothing found\n                node = null;\n            }\n        }\n\n        return this;\n    },\n\n    toBBox: function (item) { return item; },\n\n    compareMinX: function (a, b) { return a[0] - b[0]; },\n    compareMinY: function (a, b) { return a[1] - b[1]; },\n\n    toJSON: function () { return this.data; },\n\n    fromJSON: function (data) {\n        this.data = data;\n        return this;\n    },\n\n    _all: function (node, result) {\n        var nodesToSearch = [];\n        while (node) {\n            if (node.leaf) {\n                result.push.apply(result, node.children);\n            } else {\n                nodesToSearch.push.apply(nodesToSearch, node.children);\n            }\n            node = nodesToSearch.pop();\n        }\n        return result;\n    },\n\n    _build: function (items, level, height) {\n\n        var N = items.length,\n            M = this._maxEntries,\n            node;\n\n        if (N <= M) {\n            node = {\n                children: items,\n                leaf: true,\n                height: 1\n            };\n            this._calcBBox(node);\n            return node;\n        }\n\n        if (!level) {\n            // target height of the bulk-loaded tree\n            height = Math.ceil(Math.log(N) / Math.log(M));\n\n            // target number of root entries to maximize storage utilization\n            M = Math.ceil(N / Math.pow(M, height - 1));\n\n            items.sort(this.compareMinX);\n        }\n\n        // TODO eliminate recursion?\n\n        node = {\n            children: [],\n            height: height\n        };\n\n        var N1 = Math.ceil(N / M) * Math.ceil(Math.sqrt(M)),\n            N2 = Math.ceil(N / M),\n            compare = level % 2 === 1 ? this.compareMinX : this.compareMinY,\n            i, j, slice, sliceLen, childNode;\n\n        // split the items into M mostly square tiles\n        for (i = 0; i < N; i += N1) {\n            slice = items.slice(i, i + N1).sort(compare);\n\n            for (j = 0, sliceLen = slice.length; j < sliceLen; j += N2) {\n                // pack each entry recursively\n                childNode = this._build(slice.slice(j, j + N2), level + 1, height - 1);\n                node.children.push(childNode);\n            }\n        }\n\n        this._calcBBox(node);\n\n        return node;\n    },\n\n    _chooseSubtree: function (bbox, node, level, path) {\n\n        var i, len, child, targetNode, area, enlargement, minArea, minEnlargement;\n\n        while (true) {\n            path.push(node);\n\n            if (node.leaf || path.length - 1 === level) { break; }\n\n            minArea = minEnlargement = Infinity;\n\n            for (i = 0, len = node.children.length; i < len; i++) {\n                child = node.children[i];\n                area = this._area(child.bbox);\n                enlargement = this._enlargedArea(bbox, child.bbox) - area;\n\n                // choose entry with the least area enlargement\n                if (enlargement < minEnlargement) {\n                    minEnlargement = enlargement;\n                    minArea = area < minArea ? area : minArea;\n                    targetNode = child;\n\n                } else if (enlargement === minEnlargement) {\n                    // otherwise choose one with the smallest area\n                    if (area < minArea) {\n                        minArea = area;\n                        targetNode = child;\n                    }\n                }\n            }\n\n            node = targetNode;\n        }\n\n        return node;\n    },\n\n    _insert: function (item, level, isNode) {\n\n        var bbox = isNode ? item.bbox : this.toBBox(item),\n            insertPath = [];\n\n        // find the best node for accommodating the item, saving all nodes along the path too\n        var node = this._chooseSubtree(bbox, this.data, level, insertPath);\n\n        // put the item into the node\n        node.children.push(item);\n        this._extend(node.bbox, bbox);\n\n        // split on node overflow; propagate upwards if necessary\n        while (level >= 0) {\n            if (insertPath[level].children.length > this._maxEntries) {\n                this._split(insertPath, level);\n                level--;\n            } else {\n              break;\n            }\n        }\n\n        // adjust bboxes along the insertion path\n        this._adjustParentBBoxes(bbox, insertPath, level);\n    },\n\n    // split overflowed node into two\n    _split: function (insertPath, level) {\n\n        var node = insertPath[level],\n            M = node.children.length,\n            m = this._minEntries;\n\n        this._chooseSplitAxis(node, m, M);\n\n        var newNode = {\n            children: node.children.splice(this._chooseSplitIndex(node, m, M)),\n            height: node.height\n        };\n\n        if (node.leaf) {\n            newNode.leaf = true;\n        }\n\n        this._calcBBox(node);\n        this._calcBBox(newNode);\n\n        if (level) {\n            insertPath[level - 1].children.push(newNode);\n        } else {\n            this._splitRoot(node, newNode);\n        }\n    },\n\n    _splitRoot: function (node, newNode) {\n        // split root node\n        this.data = {};\n        this.data.children = [node, newNode];\n        this.data.height = node.height + 1;\n        this._calcBBox(this.data);\n    },\n\n    _chooseSplitIndex: function (node, m, M) {\n\n        var i, bbox1, bbox2, overlap, area, minOverlap, minArea, index;\n\n        minOverlap = minArea = Infinity;\n\n        for (i = m; i <= M - m; i++) {\n            bbox1 = this._distBBox(node, 0, i);\n            bbox2 = this._distBBox(node, i, M);\n\n            overlap = this._intersectionArea(bbox1, bbox2);\n            area = this._area(bbox1) + this._area(bbox2);\n\n            // choose distribution with minimum overlap\n            if (overlap < minOverlap) {\n                minOverlap = overlap;\n                index = i;\n\n                minArea = area < minArea ? area : minArea;\n\n            } else if (overlap === minOverlap) {\n                // otherwise choose distribution with minimum area\n                if (area < minArea) {\n                    minArea = area;\n                    index = i;\n                }\n            }\n        }\n\n        return index;\n    },\n\n    // sorts node children by the best axis for split\n    _chooseSplitAxis: function (node, m, M) {\n\n        var compareMinX = node.leaf ? this.compareMinX : this._compareNodeMinX,\n            compareMinY = node.leaf ? this.compareMinY : this._compareNodeMinY,\n            xMargin = this._allDistMargin(node, m, M, compareMinX),\n            yMargin = this._allDistMargin(node, m, M, compareMinY);\n\n        // if total distributions margin value is minimal for x, sort by minX,\n        // otherwise it's already sorted by minY\n\n        if (xMargin < yMargin) {\n            node.children.sort(compareMinX);\n        }\n    },\n\n    // total margin of all possible split distributions where each node is at least m full\n    _allDistMargin: function (node, m, M, compare) {\n\n        node.children.sort(compare);\n\n        var leftBBox = this._distBBox(node, 0, m),\n            rightBBox = this._distBBox(node, M - m, M),\n            margin = this._margin(leftBBox) + this._margin(rightBBox),\n            i, child;\n\n        for (i = m; i < M - m; i++) {\n            child = node.children[i];\n            this._extend(leftBBox, node.leaf ? this.toBBox(child) : child.bbox);\n            margin += this._margin(leftBBox);\n        }\n\n        for (i = M - m - 1; i >= m; i--) {\n            child = node.children[i];\n            this._extend(rightBBox, node.leaf ? this.toBBox(child) : child.bbox);\n            margin += this._margin(rightBBox);\n        }\n\n        return margin;\n    },\n\n    // min bounding rectangle of node children from k to p-1\n    _distBBox: function (node, k, p) {\n        var bbox = this._empty();\n\n        for (var i = k, child; i < p; i++) {\n            child = node.children[i];\n            this._extend(bbox, node.leaf ? this.toBBox(child) : child.bbox);\n        }\n\n        return bbox;\n    },\n\n    // calculate node's bbox from bboxes of its children\n    _calcBBox: function (node) {\n        node.bbox = this._distBBox(node, 0, node.children.length);\n    },\n\n    _adjustParentBBoxes: function (bbox, path, level) {\n        // adjust bboxes along the given tree path\n        for (var i = level; i >= 0; i--) {\n            this._extend(path[i].bbox, bbox);\n        }\n    },\n\n    _condense: function (path) {\n        // go through the path, removing empty nodes and updating bboxes\n        for (var i = path.length - 1, parent; i >= 0; i--) {\n            if (path[i].children.length === 0) {\n                if (i > 0) {\n                    parent = path[i - 1].children;\n                    parent.splice(parent.indexOf(path[i]), 1);\n                } else {\n                    this.clear();\n                }\n            } else {\n                this._calcBBox(path[i]);\n            }\n        }\n    },\n\n    _contains: function(a, b) {\n        return a[0] <= b[0] &&\n               a[1] <= b[1] &&\n               b[2] <= a[2] &&\n               b[3] <= a[3];\n    },\n\n    _intersects: function (a, b) {\n        return b[0] <= a[2] &&\n               b[1] <= a[3] &&\n               b[2] >= a[0] &&\n               b[3] >= a[1];\n    },\n\n    _extend: function (a, b) {\n        a[0] = Math.min(a[0], b[0]);\n        a[1] = Math.min(a[1], b[1]);\n        a[2] = Math.max(a[2], b[2]);\n        a[3] = Math.max(a[3], b[3]);\n        return a;\n    },\n\n    _area:   function (a) { return (a[2] - a[0]) * (a[3] - a[1]); },\n    _margin: function (a) { return (a[2] - a[0]) + (a[3] - a[1]); },\n\n    _enlargedArea: function (a, b) {\n        return (Math.max(b[2], a[2]) - Math.min(b[0], a[0])) *\n               (Math.max(b[3], a[3]) - Math.min(b[1], a[1]));\n    },\n\n    _intersectionArea: function (a, b) {\n        var minX = Math.max(a[0], b[0]),\n            minY = Math.max(a[1], b[1]),\n            maxX = Math.min(a[2], b[2]),\n            maxY = Math.min(a[3], b[3]);\n\n        return Math.max(0, maxX - minX) *\n               Math.max(0, maxY - minY);\n    },\n\n    _empty: function () { return [Infinity, Infinity, -Infinity, -Infinity]; },\n\n    _compareNodeMinX: function (a, b) { return a.bbox[0] - b.bbox[0]; },\n    _compareNodeMinY: function (a, b) { return a.bbox[1] - b.bbox[1]; },\n\n    _initFormat: function (format) {\n        // data format (minX, minY, maxX, maxY accessors)\n\n        // uses eval-type function compilation instead of just accepting a toBBox function\n        // because the algorithms are very sensitive to sorting functions performance,\n        // so they should be dead simple and without inner calls\n\n        // jshint evil: true\n\n        var compareArr = ['return a', ' - b', ';'];\n\n        this.compareMinX = new Function('a', 'b', compareArr.join(format[0]));\n        this.compareMinY = new Function('a', 'b', compareArr.join(format[1]));\n\n        this.toBBox = new Function('a', 'return [a' + format.join(', a') + '];');\n    }\n};\n\nif (typeof define === 'function' && define.amd) {\n    define(function() {\n        return rbush;\n    });\n} else if (typeof module !== 'undefined') {\n    module.exports = rbush;\n} else if (typeof self !== 'undefined') {\n    self.rbush = rbush;\n} else {\n    window.rbush = rbush;\n}\n\n})();\n","/*\n * Copyright (C) 2008 Apple Inc. All Rights Reserved.\n *\n * Redistribution and use in source and binary forms, with or without\n * modification, are permitted provided that the following conditions\n * are met:\n * 1. Redistributions of source code must retain the above copyright\n *    notice, this list of conditions and the following disclaimer.\n * 2. Redistributions in binary form must reproduce the above copyright\n *    notice, this list of conditions and the following disclaimer in the\n *    documentation and/or other materials provided with the distribution.\n *\n * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY\n * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE\n * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR\n * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR\n * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,\n * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,\n * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR\n * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY\n * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT\n * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE\n * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.\n *\n * Ported from Webkit\n * http://svn.webkit.org/repository/webkit/trunk/Source/WebCore/platform/graphics/UnitBezier.h\n */\n\nmodule.exports = UnitBezier;\n\nfunction UnitBezier(p1x, p1y, p2x, p2y) {\n    // Calculate the polynomial coefficients, implicit first and last control points are (0,0) and (1,1).\n    this.cx = 3.0 * p1x;\n    this.bx = 3.0 * (p2x - p1x) - this.cx;\n    this.ax = 1.0 - this.cx - this.bx;\n\n    this.cy = 3.0 * p1y;\n    this.by = 3.0 * (p2y - p1y) - this.cy;\n    this.ay = 1.0 - this.cy - this.by;\n\n    this.p1x = p1x;\n    this.p1y = p2y;\n    this.p2x = p2x;\n    this.p2y = p2y;\n}\n\nUnitBezier.prototype.sampleCurveX = function(t) {\n    // `ax t^3 + bx t^2 + cx t' expanded using Horner's rule.\n    return ((this.ax * t + this.bx) * t + this.cx) * t;\n};\n\nUnitBezier.prototype.sampleCurveY = function(t) {\n    return ((this.ay * t + this.by) * t + this.cy) * t;\n};\n\nUnitBezier.prototype.sampleCurveDerivativeX = function(t) {\n    return (3.0 * this.ax * t + 2.0 * this.bx) * t + this.cx;\n};\n\nUnitBezier.prototype.solveCurveX = function(x, epsilon) {\n    if (typeof epsilon === 'undefined') epsilon = 1e-6;\n\n    var t0, t1, t2, x2, i;\n\n    // First try a few iterations of Newton's method -- normally very fast.\n    for (t2 = x, i = 0; i < 8; i++) {\n\n        x2 = this.sampleCurveX(t2) - x;\n        if (Math.abs(x2) < epsilon) return t2;\n\n        var d2 = this.sampleCurveDerivativeX(t2);\n        if (Math.abs(d2) < 1e-6) break;\n\n        t2 = t2 - x2 / d2;\n    }\n\n    // Fall back to the bisection method for reliability.\n    t0 = 0.0;\n    t1 = 1.0;\n    t2 = x;\n\n    if (t2 < t0) return t0;\n    if (t2 > t1) return t1;\n\n    while (t0 < t1) {\n\n        x2 = this.sampleCurveX(t2);\n        if (Math.abs(x2 - x) < epsilon) return t2;\n\n        if (x > x2) {\n            t0 = t2;\n        } else {\n            t1 = t2;\n        }\n\n        t2 = (t1 - t0) * 0.5 + t0;\n    }\n\n    // Failure.\n    return t2;\n};\n\nUnitBezier.prototype.solve = function(x, epsilon) {\n    return this.sampleCurveY(this.solveCurveX(x, epsilon));\n};\n","module.exports.VectorTile = require('./lib/vectortile.js');\nmodule.exports.VectorTileFeature = require('./lib/vectortilefeature.js');\nmodule.exports.VectorTileLayer = require('./lib/vectortilelayer.js');\n","'use strict';\n\nvar VectorTileLayer = require('./vectortilelayer');\n\nmodule.exports = VectorTile;\n\nfunction VectorTile(buffer, end) {\n\n    this.layers = {};\n    this._buffer = buffer;\n\n    end = end || buffer.length;\n\n    while (buffer.pos < end) {\n        var val = buffer.readVarint(),\n            tag = val >> 3;\n\n        if (tag == 3) {\n            var layer = this.readLayer();\n            if (layer.length) this.layers[layer.name] = layer;\n        } else {\n            buffer.skip(val);\n        }\n    }\n}\n\nVectorTile.prototype.readLayer = function() {\n    var buffer = this._buffer,\n        bytes = buffer.readVarint(),\n        end = buffer.pos + bytes,\n        layer = new VectorTileLayer(buffer, end);\n\n    buffer.pos = end;\n\n    return layer;\n};\n","'use strict';\n\nvar Point = require('point-geometry');\n\nmodule.exports = VectorTileFeature;\n\nfunction VectorTileFeature(buffer, end, extent, keys, values) {\n\n    this.properties = {};\n\n    // Public\n    this.extent = extent;\n    this.type = 0;\n\n    // Private\n    this._buffer = buffer;\n    this._geometry = -1;\n\n    end = end || buffer.length;\n\n    while (buffer.pos < end) {\n        var val = buffer.readVarint(),\n            tag = val >> 3;\n\n        if (tag == 1) {\n            this._id = buffer.readVarint();\n\n        } else if (tag == 2) {\n            var tagEnd = buffer.pos + buffer.readVarint();\n\n            while (buffer.pos < tagEnd) {\n                var key = keys[buffer.readVarint()];\n                var value = values[buffer.readVarint()];\n                this.properties[key] = value;\n            }\n\n        } else if (tag == 3) {\n            this.type = buffer.readVarint();\n\n        } else if (tag == 4) {\n            this._geometry = buffer.pos;\n            buffer.skip(val);\n\n        } else {\n            buffer.skip(val);\n        }\n    }\n}\n\nVectorTileFeature.types = ['Unknown', 'Point', 'LineString', 'Polygon'];\n\nVectorTileFeature.prototype.loadGeometry = function() {\n    var buffer = this._buffer;\n    buffer.pos = this._geometry;\n\n    var bytes = buffer.readVarint(),\n        end = buffer.pos + bytes,\n        cmd = 1,\n        length = 0,\n        x = 0,\n        y = 0,\n        lines = [],\n        line;\n\n    while (buffer.pos < end) {\n        if (!length) {\n            var cmd_length = buffer.readVarint();\n            cmd = cmd_length & 0x7;\n            length = cmd_length >> 3;\n        }\n\n        length--;\n\n        if (cmd === 1 || cmd === 2) {\n            x += buffer.readSVarint();\n            y += buffer.readSVarint();\n\n            if (cmd === 1) {\n                // moveTo\n                if (line) {\n                    lines.push(line);\n                }\n                line = [];\n            }\n\n            line.push(new Point(x, y));\n        } else if (cmd === 7) {\n            // closePolygon\n            line.push(line[0].clone());\n        } else {\n            throw new Error('unknown command ' + cmd);\n        }\n    }\n\n    if (line) lines.push(line);\n\n    return lines;\n};\n\nVectorTileFeature.prototype.bbox = function() {\n    var buffer = this._buffer;\n    buffer.pos = this._geometry;\n\n    var bytes = buffer.readVarint(),\n        end = buffer.pos + bytes,\n\n        cmd = 1,\n        length = 0,\n        x = 0,\n        y = 0,\n        x1 = Infinity,\n        x2 = -Infinity,\n        y1 = Infinity,\n        y2 = -Infinity;\n\n    while (buffer.pos < end) {\n        if (!length) {\n            var cmd_length = buffer.readVarint();\n            cmd = cmd_length & 0x7;\n            length = cmd_length >> 3;\n        }\n\n        length--;\n\n        if (cmd === 1 || cmd === 2) {\n            x += buffer.readSVarint();\n            y += buffer.readSVarint();\n            if (x < x1) x1 = x;\n            if (x > x2) x2 = x;\n            if (y < y1) y1 = y;\n            if (y > y2) y2 = y;\n\n        } else if (cmd !== 7) {\n            throw new Error('unknown command ' + cmd);\n        }\n    }\n\n    return [x1, y1, x2, y2];\n};\n","'use strict';\n\nvar VectorTileFeature = require('./vectortilefeature.js');\n\nmodule.exports = VectorTileLayer;\nfunction VectorTileLayer(buffer, end) {\n    // Public\n    this.version = 1;\n    this.name = null;\n    this.extent = 4096;\n    this.length = 0;\n\n    // Private\n    this._buffer = buffer;\n    this._keys = [];\n    this._values = [];\n    this._features = [];\n\n    var val, tag;\n\n    end = end || buffer.length;\n\n    while (buffer.pos < end) {\n        val = buffer.readVarint();\n        tag = val >> 3;\n\n        if (tag === 15) {\n            this.version = buffer.readVarint();\n        } else if (tag === 1) {\n            this.name = buffer.readString();\n        } else if (tag === 5) {\n            this.extent = buffer.readVarint();\n        } else if (tag === 2) {\n            this.length++;\n            this._features.push(buffer.pos);\n            buffer.skip(val);\n\n        } else if (tag === 3) {\n            this._keys.push(buffer.readString());\n        } else if (tag === 4) {\n            this._values.push(this.readFeatureValue());\n        } else {\n            buffer.skip(val);\n        }\n    }\n}\n\nVectorTileLayer.prototype.readFeatureValue = function() {\n    var buffer = this._buffer,\n        value = null,\n        bytes = buffer.readVarint(),\n        end = buffer.pos + bytes,\n        val, tag;\n\n    while (buffer.pos < end) {\n        val = buffer.readVarint();\n        tag = val >> 3;\n\n        if (tag == 1) {\n            value = buffer.readString();\n        } else if (tag == 2) {\n            throw new Error('read float');\n        } else if (tag == 3) {\n            value = buffer.readDouble();\n        } else if (tag == 4) {\n            value = buffer.readVarint();\n        } else if (tag == 5) {\n            throw new Error('read uint');\n        } else if (tag == 6) {\n            value = buffer.readSVarint();\n        } else if (tag == 7) {\n            value = Boolean(buffer.readVarint());\n        } else {\n            buffer.skip(val);\n        }\n    }\n\n    return value;\n};\n\n// return feature `i` from this layer as a `VectorTileFeature`\nVectorTileLayer.prototype.feature = function(i) {\n    if (i < 0 || i >= this._features.length) throw new Error('feature index out of bounds');\n\n    this._buffer.pos = this._features[i];\n    var end = this._buffer.readVarint() + this._buffer.pos;\n\n    return new VectorTileFeature(this._buffer, end, this.extent, this._keys, this._values);\n};\n"]}