[ Index ] |
MailPress 7.2 |
[ Index ] [ Classes ] [ Functions ] [ Variables ] [ Constants ] [ Statistics ] |
[Summary view] [Print] [Text view]
1 /* @preserve 2 * Leaflet 1.6.0+Detached: 0c81bdf904d864fd12a286e3d1979f47aba17991.0c81bdf, a JS library for interactive maps. http://leafletjs.com 3 * (c) 2010-2019 Vladimir Agafonkin, (c) 2010-2011 CloudMade 4 */ 5 6 (function (global, factory) { 7 typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports) : 8 typeof define === 'function' && define.amd ? define(['exports'], factory) : 9 (factory((global.L = {}))); 10 }(this, (function (exports) { 'use strict'; 11 12 var version = "1.6.0+HEAD.0c81bdf"; 13 14 /* 15 * @namespace Util 16 * 17 * Various utility functions, used by Leaflet internally. 18 */ 19 20 var freeze = Object.freeze; 21 Object.freeze = function (obj) { return obj; }; 22 23 // @function extend(dest: Object, src?: Object): Object 24 // Merges the properties of the `src` object (or multiple objects) into `dest` object and returns the latter. Has an `L.extend` shortcut. 25 function extend(dest) { 26 var i, j, len, src; 27 28 for (j = 1, len = arguments.length; j < len; j++) { 29 src = arguments[j]; 30 for (i in src) { 31 dest[i] = src[i]; 32 } 33 } 34 return dest; 35 } 36 37 // @function create(proto: Object, properties?: Object): Object 38 // Compatibility polyfill for [Object.create](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Object/create) 39 var create = Object.create || (function () { 40 function F() {} 41 return function (proto) { 42 F.prototype = proto; 43 return new F(); 44 }; 45 })(); 46 47 // @function bind(fn: Function, …): Function 48 // Returns a new function bound to the arguments passed, like [Function.prototype.bind](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Function/bind). 49 // Has a `L.bind()` shortcut. 50 function bind(fn, obj) { 51 var slice = Array.prototype.slice; 52 53 if (fn.bind) { 54 return fn.bind.apply(fn, slice.call(arguments, 1)); 55 } 56 57 var args = slice.call(arguments, 2); 58 59 return function () { 60 return fn.apply(obj, args.length ? args.concat(slice.call(arguments)) : arguments); 61 }; 62 } 63 64 // @property lastId: Number 65 // Last unique ID used by [`stamp()`](#util-stamp) 66 var lastId = 0; 67 68 // @function stamp(obj: Object): Number 69 // Returns the unique ID of an object, assigning it one if it doesn't have it. 70 function stamp(obj) { 71 /*eslint-disable */ 72 obj._leaflet_id = obj._leaflet_id || ++lastId; 73 return obj._leaflet_id; 74 /* eslint-enable */ 75 } 76 77 // @function throttle(fn: Function, time: Number, context: Object): Function 78 // Returns a function which executes function `fn` with the given scope `context` 79 // (so that the `this` keyword refers to `context` inside `fn`'s code). The function 80 // `fn` will be called no more than one time per given amount of `time`. The arguments 81 // received by the bound function will be any arguments passed when binding the 82 // function, followed by any arguments passed when invoking the bound function. 83 // Has an `L.throttle` shortcut. 84 function throttle(fn, time, context) { 85 var lock, args, wrapperFn, later; 86 87 later = function () { 88 // reset lock and call if queued 89 lock = false; 90 if (args) { 91 wrapperFn.apply(context, args); 92 args = false; 93 } 94 }; 95 96 wrapperFn = function () { 97 if (lock) { 98 // called too soon, queue to call later 99 args = arguments; 100 101 } else { 102 // call and lock until later 103 fn.apply(context, arguments); 104 setTimeout(later, time); 105 lock = true; 106 } 107 }; 108 109 return wrapperFn; 110 } 111 112 // @function wrapNum(num: Number, range: Number[], includeMax?: Boolean): Number 113 // Returns the number `num` modulo `range` in such a way so it lies within 114 // `range[0]` and `range[1]`. The returned value will be always smaller than 115 // `range[1]` unless `includeMax` is set to `true`. 116 function wrapNum(x, range, includeMax) { 117 var max = range[1], 118 min = range[0], 119 d = max - min; 120 return x === max && includeMax ? x : ((x - min) % d + d) % d + min; 121 } 122 123 // @function falseFn(): Function 124 // Returns a function which always returns `false`. 125 function falseFn() { return false; } 126 127 // @function formatNum(num: Number, digits?: Number): Number 128 // Returns the number `num` rounded to `digits` decimals, or to 6 decimals by default. 129 function formatNum(num, digits) { 130 var pow = Math.pow(10, (digits === undefined ? 6 : digits)); 131 return Math.round(num * pow) / pow; 132 } 133 134 // @function trim(str: String): String 135 // Compatibility polyfill for [String.prototype.trim](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/String/Trim) 136 function trim(str) { 137 return str.trim ? str.trim() : str.replace(/^\s+|\s+$/g, ''); 138 } 139 140 // @function splitWords(str: String): String[] 141 // Trims and splits the string on whitespace and returns the array of parts. 142 function splitWords(str) { 143 return trim(str).split(/\s+/); 144 } 145 146 // @function setOptions(obj: Object, options: Object): Object 147 // Merges the given properties to the `options` of the `obj` object, returning the resulting options. See `Class options`. Has an `L.setOptions` shortcut. 148 function setOptions(obj, options) { 149 if (!obj.hasOwnProperty('options')) { 150 obj.options = obj.options ? create(obj.options) : {}; 151 } 152 for (var i in options) { 153 obj.options[i] = options[i]; 154 } 155 return obj.options; 156 } 157 158 // @function getParamString(obj: Object, existingUrl?: String, uppercase?: Boolean): String 159 // Converts an object into a parameter URL string, e.g. `{a: "foo", b: "bar"}` 160 // translates to `'?a=foo&b=bar'`. If `existingUrl` is set, the parameters will 161 // be appended at the end. If `uppercase` is `true`, the parameter names will 162 // be uppercased (e.g. `'?A=foo&B=bar'`) 163 function getParamString(obj, existingUrl, uppercase) { 164 var params = []; 165 for (var i in obj) { 166 params.push(encodeURIComponent(uppercase ? i.toUpperCase() : i) + '=' + encodeURIComponent(obj[i])); 167 } 168 return ((!existingUrl || existingUrl.indexOf('?') === -1) ? '?' : '&') + params.join('&'); 169 } 170 171 var templateRe = /\{ *([\w_-]+) *\}/g; 172 173 // @function template(str: String, data: Object): String 174 // Simple templating facility, accepts a template string of the form `'Hello {a}, {b}'` 175 // and a data object like `{a: 'foo', b: 'bar'}`, returns evaluated string 176 // `('Hello foo, bar')`. You can also specify functions instead of strings for 177 // data values — they will be evaluated passing `data` as an argument. 178 function template(str, data) { 179 return str.replace(templateRe, function (str, key) { 180 var value = data[key]; 181 182 if (value === undefined) { 183 throw new Error('No value provided for variable ' + str); 184 185 } else if (typeof value === 'function') { 186 value = value(data); 187 } 188 return value; 189 }); 190 } 191 192 // @function isArray(obj): Boolean 193 // Compatibility polyfill for [Array.isArray](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Array/isArray) 194 var isArray = Array.isArray || function (obj) { 195 return (Object.prototype.toString.call(obj) === '[object Array]'); 196 }; 197 198 // @function indexOf(array: Array, el: Object): Number 199 // Compatibility polyfill for [Array.prototype.indexOf](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Array/indexOf) 200 function indexOf(array, el) { 201 for (var i = 0; i < array.length; i++) { 202 if (array[i] === el) { return i; } 203 } 204 return -1; 205 } 206 207 // @property emptyImageUrl: String 208 // Data URI string containing a base64-encoded empty GIF image. 209 // Used as a hack to free memory from unused images on WebKit-powered 210 // mobile devices (by setting image `src` to this string). 211 var emptyImageUrl = ''; 212 213 // inspired by http://paulirish.com/2011/requestanimationframe-for-smart-animating/ 214 215 function getPrefixed(name) { 216 return window['webkit' + name] || window['moz' + name] || window['ms' + name]; 217 } 218 219 var lastTime = 0; 220 221 // fallback for IE 7-8 222 function timeoutDefer(fn) { 223 var time = +new Date(), 224 timeToCall = Math.max(0, 16 - (time - lastTime)); 225 226 lastTime = time + timeToCall; 227 return window.setTimeout(fn, timeToCall); 228 } 229 230 var requestFn = window.requestAnimationFrame || getPrefixed('RequestAnimationFrame') || timeoutDefer; 231 var cancelFn = window.cancelAnimationFrame || getPrefixed('CancelAnimationFrame') || 232 getPrefixed('CancelRequestAnimationFrame') || function (id) { window.clearTimeout(id); }; 233 234 // @function requestAnimFrame(fn: Function, context?: Object, immediate?: Boolean): Number 235 // Schedules `fn` to be executed when the browser repaints. `fn` is bound to 236 // `context` if given. When `immediate` is set, `fn` is called immediately if 237 // the browser doesn't have native support for 238 // [`window.requestAnimationFrame`](https://developer.mozilla.org/docs/Web/API/window/requestAnimationFrame), 239 // otherwise it's delayed. Returns a request ID that can be used to cancel the request. 240 function requestAnimFrame(fn, context, immediate) { 241 if (immediate && requestFn === timeoutDefer) { 242 fn.call(context); 243 } else { 244 return requestFn.call(window, bind(fn, context)); 245 } 246 } 247 248 // @function cancelAnimFrame(id: Number): undefined 249 // Cancels a previous `requestAnimFrame`. See also [window.cancelAnimationFrame](https://developer.mozilla.org/docs/Web/API/window/cancelAnimationFrame). 250 function cancelAnimFrame(id) { 251 if (id) { 252 cancelFn.call(window, id); 253 } 254 } 255 256 257 var Util = (Object.freeze || Object)({ 258 freeze: freeze, 259 extend: extend, 260 create: create, 261 bind: bind, 262 lastId: lastId, 263 stamp: stamp, 264 throttle: throttle, 265 wrapNum: wrapNum, 266 falseFn: falseFn, 267 formatNum: formatNum, 268 trim: trim, 269 splitWords: splitWords, 270 setOptions: setOptions, 271 getParamString: getParamString, 272 template: template, 273 isArray: isArray, 274 indexOf: indexOf, 275 emptyImageUrl: emptyImageUrl, 276 requestFn: requestFn, 277 cancelFn: cancelFn, 278 requestAnimFrame: requestAnimFrame, 279 cancelAnimFrame: cancelAnimFrame 280 }); 281 282 // @class Class 283 // @aka L.Class 284 285 // @section 286 // @uninheritable 287 288 // Thanks to John Resig and Dean Edwards for inspiration! 289 290 function Class() {} 291 292 Class.extend = function (props) { 293 294 // @function extend(props: Object): Function 295 // [Extends the current class](#class-inheritance) given the properties to be included. 296 // Returns a Javascript function that is a class constructor (to be called with `new`). 297 var NewClass = function () { 298 299 // call the constructor 300 if (this.initialize) { 301 this.initialize.apply(this, arguments); 302 } 303 304 // call all constructor hooks 305 this.callInitHooks(); 306 }; 307 308 var parentProto = NewClass.__super__ = this.prototype; 309 310 var proto = create(parentProto); 311 proto.constructor = NewClass; 312 313 NewClass.prototype = proto; 314 315 // inherit parent's statics 316 for (var i in this) { 317 if (this.hasOwnProperty(i) && i !== 'prototype' && i !== '__super__') { 318 NewClass[i] = this[i]; 319 } 320 } 321 322 // mix static properties into the class 323 if (props.statics) { 324 extend(NewClass, props.statics); 325 delete props.statics; 326 } 327 328 // mix includes into the prototype 329 if (props.includes) { 330 checkDeprecatedMixinEvents(props.includes); 331 extend.apply(null, [proto].concat(props.includes)); 332 delete props.includes; 333 } 334 335 // merge options 336 if (proto.options) { 337 props.options = extend(create(proto.options), props.options); 338 } 339 340 // mix given properties into the prototype 341 extend(proto, props); 342 343 proto._initHooks = []; 344 345 // add method for calling all hooks 346 proto.callInitHooks = function () { 347 348 if (this._initHooksCalled) { return; } 349 350 if (parentProto.callInitHooks) { 351 parentProto.callInitHooks.call(this); 352 } 353 354 this._initHooksCalled = true; 355 356 for (var i = 0, len = proto._initHooks.length; i < len; i++) { 357 proto._initHooks[i].call(this); 358 } 359 }; 360 361 return NewClass; 362 }; 363 364 365 // @function include(properties: Object): this 366 // [Includes a mixin](#class-includes) into the current class. 367 Class.include = function (props) { 368 extend(this.prototype, props); 369 return this; 370 }; 371 372 // @function mergeOptions(options: Object): this 373 // [Merges `options`](#class-options) into the defaults of the class. 374 Class.mergeOptions = function (options) { 375 extend(this.prototype.options, options); 376 return this; 377 }; 378 379 // @function addInitHook(fn: Function): this 380 // Adds a [constructor hook](#class-constructor-hooks) to the class. 381 Class.addInitHook = function (fn) { // (Function) || (String, args...) 382 var args = Array.prototype.slice.call(arguments, 1); 383 384 var init = typeof fn === 'function' ? fn : function () { 385 this[fn].apply(this, args); 386 }; 387 388 this.prototype._initHooks = this.prototype._initHooks || []; 389 this.prototype._initHooks.push(init); 390 return this; 391 }; 392 393 function checkDeprecatedMixinEvents(includes) { 394 if (typeof L === 'undefined' || !L || !L.Mixin) { return; } 395 396 includes = isArray(includes) ? includes : [includes]; 397 398 for (var i = 0; i < includes.length; i++) { 399 if (includes[i] === L.Mixin.Events) { 400 console.warn('Deprecated include of L.Mixin.Events: ' + 401 'this property will be removed in future releases, ' + 402 'please inherit from L.Evented instead.', new Error().stack); 403 } 404 } 405 } 406 407 /* 408 * @class Evented 409 * @aka L.Evented 410 * @inherits Class 411 * 412 * A set of methods shared between event-powered classes (like `Map` and `Marker`). Generally, events allow you to execute some function when something happens with an object (e.g. the user clicks on the map, causing the map to fire `'click'` event). 413 * 414 * @example 415 * 416 * ```js 417 * map.on('click', function(e) { 418 * alert(e.latlng); 419 * } ); 420 * ``` 421 * 422 * Leaflet deals with event listeners by reference, so if you want to add a listener and then remove it, define it as a function: 423 * 424 * ```js 425 * function onClick(e) { ... } 426 * 427 * map.on('click', onClick); 428 * map.off('click', onClick); 429 * ``` 430 */ 431 432 var Events = { 433 /* @method on(type: String, fn: Function, context?: Object): this 434 * Adds a listener function (`fn`) to a particular event type of the object. You can optionally specify the context of the listener (object the this keyword will point to). You can also pass several space-separated types (e.g. `'click dblclick'`). 435 * 436 * @alternative 437 * @method on(eventMap: Object): this 438 * Adds a set of type/listener pairs, e.g. `{click: onClick, mousemove: onMouseMove}` 439 */ 440 on: function (types, fn, context) { 441 442 // types can be a map of types/handlers 443 if (typeof types === 'object') { 444 for (var type in types) { 445 // we don't process space-separated events here for performance; 446 // it's a hot path since Layer uses the on(obj) syntax 447 this._on(type, types[type], fn); 448 } 449 450 } else { 451 // types can be a string of space-separated words 452 types = splitWords(types); 453 454 for (var i = 0, len = types.length; i < len; i++) { 455 this._on(types[i], fn, context); 456 } 457 } 458 459 return this; 460 }, 461 462 /* @method off(type: String, fn?: Function, context?: Object): this 463 * Removes a previously added listener function. If no function is specified, it will remove all the listeners of that particular event from the object. Note that if you passed a custom context to `on`, you must pass the same context to `off` in order to remove the listener. 464 * 465 * @alternative 466 * @method off(eventMap: Object): this 467 * Removes a set of type/listener pairs. 468 * 469 * @alternative 470 * @method off: this 471 * Removes all listeners to all events on the object. This includes implicitly attached events. 472 */ 473 off: function (types, fn, context) { 474 475 if (!types) { 476 // clear all listeners if called without arguments 477 delete this._events; 478 479 } else if (typeof types === 'object') { 480 for (var type in types) { 481 this._off(type, types[type], fn); 482 } 483 484 } else { 485 types = splitWords(types); 486 487 for (var i = 0, len = types.length; i < len; i++) { 488 this._off(types[i], fn, context); 489 } 490 } 491 492 return this; 493 }, 494 495 // attach listener (without syntactic sugar now) 496 _on: function (type, fn, context) { 497 this._events = this._events || {}; 498 499 /* get/init listeners for type */ 500 var typeListeners = this._events[type]; 501 if (!typeListeners) { 502 typeListeners = []; 503 this._events[type] = typeListeners; 504 } 505 506 if (context === this) { 507 // Less memory footprint. 508 context = undefined; 509 } 510 var newListener = {fn: fn, ctx: context}, 511 listeners = typeListeners; 512 513 // check if fn already there 514 for (var i = 0, len = listeners.length; i < len; i++) { 515 if (listeners[i].fn === fn && listeners[i].ctx === context) { 516 return; 517 } 518 } 519 520 listeners.push(newListener); 521 }, 522 523 _off: function (type, fn, context) { 524 var listeners, 525 i, 526 len; 527 528 if (!this._events) { return; } 529 530 listeners = this._events[type]; 531 532 if (!listeners) { 533 return; 534 } 535 536 if (!fn) { 537 // Set all removed listeners to noop so they are not called if remove happens in fire 538 for (i = 0, len = listeners.length; i < len; i++) { 539 listeners[i].fn = falseFn; 540 } 541 // clear all listeners for a type if function isn't specified 542 delete this._events[type]; 543 return; 544 } 545 546 if (context === this) { 547 context = undefined; 548 } 549 550 if (listeners) { 551 552 // find fn and remove it 553 for (i = 0, len = listeners.length; i < len; i++) { 554 var l = listeners[i]; 555 if (l.ctx !== context) { continue; } 556 if (l.fn === fn) { 557 558 // set the removed listener to noop so that's not called if remove happens in fire 559 l.fn = falseFn; 560 561 if (this._firingCount) { 562 /* copy array in case events are being fired */ 563 this._events[type] = listeners = listeners.slice(); 564 } 565 listeners.splice(i, 1); 566 567 return; 568 } 569 } 570 } 571 }, 572 573 // @method fire(type: String, data?: Object, propagate?: Boolean): this 574 // Fires an event of the specified type. You can optionally provide an data 575 // object — the first argument of the listener function will contain its 576 // properties. The event can optionally be propagated to event parents. 577 fire: function (type, data, propagate) { 578 if (!this.listens(type, propagate)) { return this; } 579 580 var event = extend({}, data, { 581 type: type, 582 target: this, 583 sourceTarget: data && data.sourceTarget || this 584 }); 585 586 if (this._events) { 587 var listeners = this._events[type]; 588 589 if (listeners) { 590 this._firingCount = (this._firingCount + 1) || 1; 591 for (var i = 0, len = listeners.length; i < len; i++) { 592 var l = listeners[i]; 593 l.fn.call(l.ctx || this, event); 594 } 595 596 this._firingCount--; 597 } 598 } 599 600 if (propagate) { 601 // propagate the event to parents (set with addEventParent) 602 this._propagateEvent(event); 603 } 604 605 return this; 606 }, 607 608 // @method listens(type: String): Boolean 609 // Returns `true` if a particular event type has any listeners attached to it. 610 listens: function (type, propagate) { 611 var listeners = this._events && this._events[type]; 612 if (listeners && listeners.length) { return true; } 613 614 if (propagate) { 615 // also check parents for listeners if event propagates 616 for (var id in this._eventParents) { 617 if (this._eventParents[id].listens(type, propagate)) { return true; } 618 } 619 } 620 return false; 621 }, 622 623 // @method once(…): this 624 // Behaves as [`on(…)`](#evented-on), except the listener will only get fired once and then removed. 625 once: function (types, fn, context) { 626 627 if (typeof types === 'object') { 628 for (var type in types) { 629 this.once(type, types[type], fn); 630 } 631 return this; 632 } 633 634 var handler = bind(function () { 635 this 636 .off(types, fn, context) 637 .off(types, handler, context); 638 }, this); 639 640 // add a listener that's executed once and removed after that 641 return this 642 .on(types, fn, context) 643 .on(types, handler, context); 644 }, 645 646 // @method addEventParent(obj: Evented): this 647 // Adds an event parent - an `Evented` that will receive propagated events 648 addEventParent: function (obj) { 649 this._eventParents = this._eventParents || {}; 650 this._eventParents[stamp(obj)] = obj; 651 return this; 652 }, 653 654 // @method removeEventParent(obj: Evented): this 655 // Removes an event parent, so it will stop receiving propagated events 656 removeEventParent: function (obj) { 657 if (this._eventParents) { 658 delete this._eventParents[stamp(obj)]; 659 } 660 return this; 661 }, 662 663 _propagateEvent: function (e) { 664 for (var id in this._eventParents) { 665 this._eventParents[id].fire(e.type, extend({ 666 layer: e.target, 667 propagatedFrom: e.target 668 }, e), true); 669 } 670 } 671 }; 672 673 // aliases; we should ditch those eventually 674 675 // @method addEventListener(…): this 676 // Alias to [`on(…)`](#evented-on) 677 Events.addEventListener = Events.on; 678 679 // @method removeEventListener(…): this 680 // Alias to [`off(…)`](#evented-off) 681 682 // @method clearAllEventListeners(…): this 683 // Alias to [`off()`](#evented-off) 684 Events.removeEventListener = Events.clearAllEventListeners = Events.off; 685 686 // @method addOneTimeEventListener(…): this 687 // Alias to [`once(…)`](#evented-once) 688 Events.addOneTimeEventListener = Events.once; 689 690 // @method fireEvent(…): this 691 // Alias to [`fire(…)`](#evented-fire) 692 Events.fireEvent = Events.fire; 693 694 // @method hasEventListeners(…): Boolean 695 // Alias to [`listens(…)`](#evented-listens) 696 Events.hasEventListeners = Events.listens; 697 698 var Evented = Class.extend(Events); 699 700 /* 701 * @class Point 702 * @aka L.Point 703 * 704 * Represents a point with `x` and `y` coordinates in pixels. 705 * 706 * @example 707 * 708 * ```js 709 * var point = L.point(200, 300); 710 * ``` 711 * 712 * All Leaflet methods and options that accept `Point` objects also accept them in a simple Array form (unless noted otherwise), so these lines are equivalent: 713 * 714 * ```js 715 * map.panBy([200, 300]); 716 * map.panBy(L.point(200, 300)); 717 * ``` 718 * 719 * Note that `Point` does not inherit from Leafet's `Class` object, 720 * which means new classes can't inherit from it, and new methods 721 * can't be added to it with the `include` function. 722 */ 723 724 function Point(x, y, round) { 725 // @property x: Number; The `x` coordinate of the point 726 this.x = (round ? Math.round(x) : x); 727 // @property y: Number; The `y` coordinate of the point 728 this.y = (round ? Math.round(y) : y); 729 } 730 731 var trunc = Math.trunc || function (v) { 732 return v > 0 ? Math.floor(v) : Math.ceil(v); 733 }; 734 735 Point.prototype = { 736 737 // @method clone(): Point 738 // Returns a copy of the current point. 739 clone: function () { 740 return new Point(this.x, this.y); 741 }, 742 743 // @method add(otherPoint: Point): Point 744 // Returns the result of addition of the current and the given points. 745 add: function (point) { 746 // non-destructive, returns a new point 747 return this.clone()._add(toPoint(point)); 748 }, 749 750 _add: function (point) { 751 // destructive, used directly for performance in situations where it's safe to modify existing point 752 this.x += point.x; 753 this.y += point.y; 754 return this; 755 }, 756 757 // @method subtract(otherPoint: Point): Point 758 // Returns the result of subtraction of the given point from the current. 759 subtract: function (point) { 760 return this.clone()._subtract(toPoint(point)); 761 }, 762 763 _subtract: function (point) { 764 this.x -= point.x; 765 this.y -= point.y; 766 return this; 767 }, 768 769 // @method divideBy(num: Number): Point 770 // Returns the result of division of the current point by the given number. 771 divideBy: function (num) { 772 return this.clone()._divideBy(num); 773 }, 774 775 _divideBy: function (num) { 776 this.x /= num; 777 this.y /= num; 778 return this; 779 }, 780 781 // @method multiplyBy(num: Number): Point 782 // Returns the result of multiplication of the current point by the given number. 783 multiplyBy: function (num) { 784 return this.clone()._multiplyBy(num); 785 }, 786 787 _multiplyBy: function (num) { 788 this.x *= num; 789 this.y *= num; 790 return this; 791 }, 792 793 // @method scaleBy(scale: Point): Point 794 // Multiply each coordinate of the current point by each coordinate of 795 // `scale`. In linear algebra terms, multiply the point by the 796 // [scaling matrix](https://en.wikipedia.org/wiki/Scaling_%28geometry%29#Matrix_representation) 797 // defined by `scale`. 798 scaleBy: function (point) { 799 return new Point(this.x * point.x, this.y * point.y); 800 }, 801 802 // @method unscaleBy(scale: Point): Point 803 // Inverse of `scaleBy`. Divide each coordinate of the current point by 804 // each coordinate of `scale`. 805 unscaleBy: function (point) { 806 return new Point(this.x / point.x, this.y / point.y); 807 }, 808 809 // @method round(): Point 810 // Returns a copy of the current point with rounded coordinates. 811 round: function () { 812 return this.clone()._round(); 813 }, 814 815 _round: function () { 816 this.x = Math.round(this.x); 817 this.y = Math.round(this.y); 818 return this; 819 }, 820 821 // @method floor(): Point 822 // Returns a copy of the current point with floored coordinates (rounded down). 823 floor: function () { 824 return this.clone()._floor(); 825 }, 826 827 _floor: function () { 828 this.x = Math.floor(this.x); 829 this.y = Math.floor(this.y); 830 return this; 831 }, 832 833 // @method ceil(): Point 834 // Returns a copy of the current point with ceiled coordinates (rounded up). 835 ceil: function () { 836 return this.clone()._ceil(); 837 }, 838 839 _ceil: function () { 840 this.x = Math.ceil(this.x); 841 this.y = Math.ceil(this.y); 842 return this; 843 }, 844 845 // @method trunc(): Point 846 // Returns a copy of the current point with truncated coordinates (rounded towards zero). 847 trunc: function () { 848 return this.clone()._trunc(); 849 }, 850 851 _trunc: function () { 852 this.x = trunc(this.x); 853 this.y = trunc(this.y); 854 return this; 855 }, 856 857 // @method distanceTo(otherPoint: Point): Number 858 // Returns the cartesian distance between the current and the given points. 859 distanceTo: function (point) { 860 point = toPoint(point); 861 862 var x = point.x - this.x, 863 y = point.y - this.y; 864 865 return Math.sqrt(x * x + y * y); 866 }, 867 868 // @method equals(otherPoint: Point): Boolean 869 // Returns `true` if the given point has the same coordinates. 870 equals: function (point) { 871 point = toPoint(point); 872 873 return point.x === this.x && 874 point.y === this.y; 875 }, 876 877 // @method contains(otherPoint: Point): Boolean 878 // Returns `true` if both coordinates of the given point are less than the corresponding current point coordinates (in absolute values). 879 contains: function (point) { 880 point = toPoint(point); 881 882 return Math.abs(point.x) <= Math.abs(this.x) && 883 Math.abs(point.y) <= Math.abs(this.y); 884 }, 885 886 // @method toString(): String 887 // Returns a string representation of the point for debugging purposes. 888 toString: function () { 889 return 'Point(' + 890 formatNum(this.x) + ', ' + 891 formatNum(this.y) + ')'; 892 } 893 }; 894 895 // @factory L.point(x: Number, y: Number, round?: Boolean) 896 // Creates a Point object with the given `x` and `y` coordinates. If optional `round` is set to true, rounds the `x` and `y` values. 897 898 // @alternative 899 // @factory L.point(coords: Number[]) 900 // Expects an array of the form `[x, y]` instead. 901 902 // @alternative 903 // @factory L.point(coords: Object) 904 // Expects a plain object of the form `{x: Number, y: Number}` instead. 905 function toPoint(x, y, round) { 906 if (x instanceof Point) { 907 return x; 908 } 909 if (isArray(x)) { 910 return new Point(x[0], x[1]); 911 } 912 if (x === undefined || x === null) { 913 return x; 914 } 915 if (typeof x === 'object' && 'x' in x && 'y' in x) { 916 return new Point(x.x, x.y); 917 } 918 return new Point(x, y, round); 919 } 920 921 /* 922 * @class Bounds 923 * @aka L.Bounds 924 * 925 * Represents a rectangular area in pixel coordinates. 926 * 927 * @example 928 * 929 * ```js 930 * var p1 = L.point(10, 10), 931 * p2 = L.point(40, 60), 932 * bounds = L.bounds(p1, p2); 933 * ``` 934 * 935 * All Leaflet methods that accept `Bounds` objects also accept them in a simple Array form (unless noted otherwise), so the bounds example above can be passed like this: 936 * 937 * ```js 938 * otherBounds.intersects([[10, 10], [40, 60]]); 939 * ``` 940 * 941 * Note that `Bounds` does not inherit from Leafet's `Class` object, 942 * which means new classes can't inherit from it, and new methods 943 * can't be added to it with the `include` function. 944 */ 945 946 function Bounds(a, b) { 947 if (!a) { return; } 948 949 var points = b ? [a, b] : a; 950 951 for (var i = 0, len = points.length; i < len; i++) { 952 this.extend(points[i]); 953 } 954 } 955 956 Bounds.prototype = { 957 // @method extend(point: Point): this 958 // Extends the bounds to contain the given point. 959 extend: function (point) { // (Point) 960 point = toPoint(point); 961 962 // @property min: Point 963 // The top left corner of the rectangle. 964 // @property max: Point 965 // The bottom right corner of the rectangle. 966 if (!this.min && !this.max) { 967 this.min = point.clone(); 968 this.max = point.clone(); 969 } else { 970 this.min.x = Math.min(point.x, this.min.x); 971 this.max.x = Math.max(point.x, this.max.x); 972 this.min.y = Math.min(point.y, this.min.y); 973 this.max.y = Math.max(point.y, this.max.y); 974 } 975 return this; 976 }, 977 978 // @method getCenter(round?: Boolean): Point 979 // Returns the center point of the bounds. 980 getCenter: function (round) { 981 return new Point( 982 (this.min.x + this.max.x) / 2, 983 (this.min.y + this.max.y) / 2, round); 984 }, 985 986 // @method getBottomLeft(): Point 987 // Returns the bottom-left point of the bounds. 988 getBottomLeft: function () { 989 return new Point(this.min.x, this.max.y); 990 }, 991 992 // @method getTopRight(): Point 993 // Returns the top-right point of the bounds. 994 getTopRight: function () { // -> Point 995 return new Point(this.max.x, this.min.y); 996 }, 997 998 // @method getTopLeft(): Point 999 // Returns the top-left point of the bounds (i.e. [`this.min`](#bounds-min)). 1000 getTopLeft: function () { 1001 return this.min; // left, top 1002 }, 1003 1004 // @method getBottomRight(): Point 1005 // Returns the bottom-right point of the bounds (i.e. [`this.max`](#bounds-max)). 1006 getBottomRight: function () { 1007 return this.max; // right, bottom 1008 }, 1009 1010 // @method getSize(): Point 1011 // Returns the size of the given bounds 1012 getSize: function () { 1013 return this.max.subtract(this.min); 1014 }, 1015 1016 // @method contains(otherBounds: Bounds): Boolean 1017 // Returns `true` if the rectangle contains the given one. 1018 // @alternative 1019 // @method contains(point: Point): Boolean 1020 // Returns `true` if the rectangle contains the given point. 1021 contains: function (obj) { 1022 var min, max; 1023 1024 if (typeof obj[0] === 'number' || obj instanceof Point) { 1025 obj = toPoint(obj); 1026 } else { 1027 obj = toBounds(obj); 1028 } 1029 1030 if (obj instanceof Bounds) { 1031 min = obj.min; 1032 max = obj.max; 1033 } else { 1034 min = max = obj; 1035 } 1036 1037 return (min.x >= this.min.x) && 1038 (max.x <= this.max.x) && 1039 (min.y >= this.min.y) && 1040 (max.y <= this.max.y); 1041 }, 1042 1043 // @method intersects(otherBounds: Bounds): Boolean 1044 // Returns `true` if the rectangle intersects the given bounds. Two bounds 1045 // intersect if they have at least one point in common. 1046 intersects: function (bounds) { // (Bounds) -> Boolean 1047 bounds = toBounds(bounds); 1048 1049 var min = this.min, 1050 max = this.max, 1051 min2 = bounds.min, 1052 max2 = bounds.max, 1053 xIntersects = (max2.x >= min.x) && (min2.x <= max.x), 1054 yIntersects = (max2.y >= min.y) && (min2.y <= max.y); 1055 1056 return xIntersects && yIntersects; 1057 }, 1058 1059 // @method overlaps(otherBounds: Bounds): Boolean 1060 // Returns `true` if the rectangle overlaps the given bounds. Two bounds 1061 // overlap if their intersection is an area. 1062 overlaps: function (bounds) { // (Bounds) -> Boolean 1063 bounds = toBounds(bounds); 1064 1065 var min = this.min, 1066 max = this.max, 1067 min2 = bounds.min, 1068 max2 = bounds.max, 1069 xOverlaps = (max2.x > min.x) && (min2.x < max.x), 1070 yOverlaps = (max2.y > min.y) && (min2.y < max.y); 1071 1072 return xOverlaps && yOverlaps; 1073 }, 1074 1075 isValid: function () { 1076 return !!(this.min && this.max); 1077 } 1078 }; 1079 1080 1081 // @factory L.bounds(corner1: Point, corner2: Point) 1082 // Creates a Bounds object from two corners coordinate pairs. 1083 // @alternative 1084 // @factory L.bounds(points: Point[]) 1085 // Creates a Bounds object from the given array of points. 1086 function toBounds(a, b) { 1087 if (!a || a instanceof Bounds) { 1088 return a; 1089 } 1090 return new Bounds(a, b); 1091 } 1092 1093 /* 1094 * @class LatLngBounds 1095 * @aka L.LatLngBounds 1096 * 1097 * Represents a rectangular geographical area on a map. 1098 * 1099 * @example 1100 * 1101 * ```js 1102 * var corner1 = L.latLng(40.712, -74.227), 1103 * corner2 = L.latLng(40.774, -74.125), 1104 * bounds = L.latLngBounds(corner1, corner2); 1105 * ``` 1106 * 1107 * All Leaflet methods that accept LatLngBounds objects also accept them in a simple Array form (unless noted otherwise), so the bounds example above can be passed like this: 1108 * 1109 * ```js 1110 * map.fitBounds([ 1111 * [40.712, -74.227], 1112 * [40.774, -74.125] 1113 * ]); 1114 * ``` 1115 * 1116 * Caution: if the area crosses the antimeridian (often confused with the International Date Line), you must specify corners _outside_ the [-180, 180] degrees longitude range. 1117 * 1118 * Note that `LatLngBounds` does not inherit from Leafet's `Class` object, 1119 * which means new classes can't inherit from it, and new methods 1120 * can't be added to it with the `include` function. 1121 */ 1122 1123 function LatLngBounds(corner1, corner2) { // (LatLng, LatLng) or (LatLng[]) 1124 if (!corner1) { return; } 1125 1126 var latlngs = corner2 ? [corner1, corner2] : corner1; 1127 1128 for (var i = 0, len = latlngs.length; i < len; i++) { 1129 this.extend(latlngs[i]); 1130 } 1131 } 1132 1133 LatLngBounds.prototype = { 1134 1135 // @method extend(latlng: LatLng): this 1136 // Extend the bounds to contain the given point 1137 1138 // @alternative 1139 // @method extend(otherBounds: LatLngBounds): this 1140 // Extend the bounds to contain the given bounds 1141 extend: function (obj) { 1142 var sw = this._southWest, 1143 ne = this._northEast, 1144 sw2, ne2; 1145 1146 if (obj instanceof LatLng) { 1147 sw2 = obj; 1148 ne2 = obj; 1149 1150 } else if (obj instanceof LatLngBounds) { 1151 sw2 = obj._southWest; 1152 ne2 = obj._northEast; 1153 1154 if (!sw2 || !ne2) { return this; } 1155 1156 } else { 1157 return obj ? this.extend(toLatLng(obj) || toLatLngBounds(obj)) : this; 1158 } 1159 1160 if (!sw && !ne) { 1161 this._southWest = new LatLng(sw2.lat, sw2.lng); 1162 this._northEast = new LatLng(ne2.lat, ne2.lng); 1163 } else { 1164 sw.lat = Math.min(sw2.lat, sw.lat); 1165 sw.lng = Math.min(sw2.lng, sw.lng); 1166 ne.lat = Math.max(ne2.lat, ne.lat); 1167 ne.lng = Math.max(ne2.lng, ne.lng); 1168 } 1169 1170 return this; 1171 }, 1172 1173 // @method pad(bufferRatio: Number): LatLngBounds 1174 // Returns bounds created by extending or retracting the current bounds by a given ratio in each direction. 1175 // For example, a ratio of 0.5 extends the bounds by 50% in each direction. 1176 // Negative values will retract the bounds. 1177 pad: function (bufferRatio) { 1178 var sw = this._southWest, 1179 ne = this._northEast, 1180 heightBuffer = Math.abs(sw.lat - ne.lat) * bufferRatio, 1181 widthBuffer = Math.abs(sw.lng - ne.lng) * bufferRatio; 1182 1183 return new LatLngBounds( 1184 new LatLng(sw.lat - heightBuffer, sw.lng - widthBuffer), 1185 new LatLng(ne.lat + heightBuffer, ne.lng + widthBuffer)); 1186 }, 1187 1188 // @method getCenter(): LatLng 1189 // Returns the center point of the bounds. 1190 getCenter: function () { 1191 return new LatLng( 1192 (this._southWest.lat + this._northEast.lat) / 2, 1193 (this._southWest.lng + this._northEast.lng) / 2); 1194 }, 1195 1196 // @method getSouthWest(): LatLng 1197 // Returns the south-west point of the bounds. 1198 getSouthWest: function () { 1199 return this._southWest; 1200 }, 1201 1202 // @method getNorthEast(): LatLng 1203 // Returns the north-east point of the bounds. 1204 getNorthEast: function () { 1205 return this._northEast; 1206 }, 1207 1208 // @method getNorthWest(): LatLng 1209 // Returns the north-west point of the bounds. 1210 getNorthWest: function () { 1211 return new LatLng(this.getNorth(), this.getWest()); 1212 }, 1213 1214 // @method getSouthEast(): LatLng 1215 // Returns the south-east point of the bounds. 1216 getSouthEast: function () { 1217 return new LatLng(this.getSouth(), this.getEast()); 1218 }, 1219 1220 // @method getWest(): Number 1221 // Returns the west longitude of the bounds 1222 getWest: function () { 1223 return this._southWest.lng; 1224 }, 1225 1226 // @method getSouth(): Number 1227 // Returns the south latitude of the bounds 1228 getSouth: function () { 1229 return this._southWest.lat; 1230 }, 1231 1232 // @method getEast(): Number 1233 // Returns the east longitude of the bounds 1234 getEast: function () { 1235 return this._northEast.lng; 1236 }, 1237 1238 // @method getNorth(): Number 1239 // Returns the north latitude of the bounds 1240 getNorth: function () { 1241 return this._northEast.lat; 1242 }, 1243 1244 // @method contains(otherBounds: LatLngBounds): Boolean 1245 // Returns `true` if the rectangle contains the given one. 1246 1247 // @alternative 1248 // @method contains (latlng: LatLng): Boolean 1249 // Returns `true` if the rectangle contains the given point. 1250 contains: function (obj) { // (LatLngBounds) or (LatLng) -> Boolean 1251 if (typeof obj[0] === 'number' || obj instanceof LatLng || 'lat' in obj) { 1252 obj = toLatLng(obj); 1253 } else { 1254 obj = toLatLngBounds(obj); 1255 } 1256 1257 var sw = this._southWest, 1258 ne = this._northEast, 1259 sw2, ne2; 1260 1261 if (obj instanceof LatLngBounds) { 1262 sw2 = obj.getSouthWest(); 1263 ne2 = obj.getNorthEast(); 1264 } else { 1265 sw2 = ne2 = obj; 1266 } 1267 1268 return (sw2.lat >= sw.lat) && (ne2.lat <= ne.lat) && 1269 (sw2.lng >= sw.lng) && (ne2.lng <= ne.lng); 1270 }, 1271 1272 // @method intersects(otherBounds: LatLngBounds): Boolean 1273 // Returns `true` if the rectangle intersects the given bounds. Two bounds intersect if they have at least one point in common. 1274 intersects: function (bounds) { 1275 bounds = toLatLngBounds(bounds); 1276 1277 var sw = this._southWest, 1278 ne = this._northEast, 1279 sw2 = bounds.getSouthWest(), 1280 ne2 = bounds.getNorthEast(), 1281 1282 latIntersects = (ne2.lat >= sw.lat) && (sw2.lat <= ne.lat), 1283 lngIntersects = (ne2.lng >= sw.lng) && (sw2.lng <= ne.lng); 1284 1285 return latIntersects && lngIntersects; 1286 }, 1287 1288 // @method overlaps(otherBounds: Bounds): Boolean 1289 // Returns `true` if the rectangle overlaps the given bounds. Two bounds overlap if their intersection is an area. 1290 overlaps: function (bounds) { 1291 bounds = toLatLngBounds(bounds); 1292 1293 var sw = this._southWest, 1294 ne = this._northEast, 1295 sw2 = bounds.getSouthWest(), 1296 ne2 = bounds.getNorthEast(), 1297 1298 latOverlaps = (ne2.lat > sw.lat) && (sw2.lat < ne.lat), 1299 lngOverlaps = (ne2.lng > sw.lng) && (sw2.lng < ne.lng); 1300 1301 return latOverlaps && lngOverlaps; 1302 }, 1303 1304 // @method toBBoxString(): String 1305 // Returns a string with bounding box coordinates in a 'southwest_lng,southwest_lat,northeast_lng,northeast_lat' format. Useful for sending requests to web services that return geo data. 1306 toBBoxString: function () { 1307 return [this.getWest(), this.getSouth(), this.getEast(), this.getNorth()].join(','); 1308 }, 1309 1310 // @method equals(otherBounds: LatLngBounds, maxMargin?: Number): Boolean 1311 // Returns `true` if the rectangle is equivalent (within a small margin of error) to the given bounds. The margin of error can be overridden by setting `maxMargin` to a small number. 1312 equals: function (bounds, maxMargin) { 1313 if (!bounds) { return false; } 1314 1315 bounds = toLatLngBounds(bounds); 1316 1317 return this._southWest.equals(bounds.getSouthWest(), maxMargin) && 1318 this._northEast.equals(bounds.getNorthEast(), maxMargin); 1319 }, 1320 1321 // @method isValid(): Boolean 1322 // Returns `true` if the bounds are properly initialized. 1323 isValid: function () { 1324 return !!(this._southWest && this._northEast); 1325 } 1326 }; 1327 1328 // TODO International date line? 1329 1330 // @factory L.latLngBounds(corner1: LatLng, corner2: LatLng) 1331 // Creates a `LatLngBounds` object by defining two diagonally opposite corners of the rectangle. 1332 1333 // @alternative 1334 // @factory L.latLngBounds(latlngs: LatLng[]) 1335 // Creates a `LatLngBounds` object defined by the geographical points it contains. Very useful for zooming the map to fit a particular set of locations with [`fitBounds`](#map-fitbounds). 1336 function toLatLngBounds(a, b) { 1337 if (a instanceof LatLngBounds) { 1338 return a; 1339 } 1340 return new LatLngBounds(a, b); 1341 } 1342 1343 /* @class LatLng 1344 * @aka L.LatLng 1345 * 1346 * Represents a geographical point with a certain latitude and longitude. 1347 * 1348 * @example 1349 * 1350 * ``` 1351 * var latlng = L.latLng(50.5, 30.5); 1352 * ``` 1353 * 1354 * All Leaflet methods that accept LatLng objects also accept them in a simple Array form and simple object form (unless noted otherwise), so these lines are equivalent: 1355 * 1356 * ``` 1357 * map.panTo([50, 30]); 1358 * map.panTo({lon: 30, lat: 50}); 1359 * map.panTo({lat: 50, lng: 30}); 1360 * map.panTo(L.latLng(50, 30)); 1361 * ``` 1362 * 1363 * Note that `LatLng` does not inherit from Leaflet's `Class` object, 1364 * which means new classes can't inherit from it, and new methods 1365 * can't be added to it with the `include` function. 1366 */ 1367 1368 function LatLng(lat, lng, alt) { 1369 if (isNaN(lat) || isNaN(lng)) { 1370 throw new Error('Invalid LatLng object: (' + lat + ', ' + lng + ')'); 1371 } 1372 1373 // @property lat: Number 1374 // Latitude in degrees 1375 this.lat = +lat; 1376 1377 // @property lng: Number 1378 // Longitude in degrees 1379 this.lng = +lng; 1380 1381 // @property alt: Number 1382 // Altitude in meters (optional) 1383 if (alt !== undefined) { 1384 this.alt = +alt; 1385 } 1386 } 1387 1388 LatLng.prototype = { 1389 // @method equals(otherLatLng: LatLng, maxMargin?: Number): Boolean 1390 // Returns `true` if the given `LatLng` point is at the same position (within a small margin of error). The margin of error can be overridden by setting `maxMargin` to a small number. 1391 equals: function (obj, maxMargin) { 1392 if (!obj) { return false; } 1393 1394 obj = toLatLng(obj); 1395 1396 var margin = Math.max( 1397 Math.abs(this.lat - obj.lat), 1398 Math.abs(this.lng - obj.lng)); 1399 1400 return margin <= (maxMargin === undefined ? 1.0E-9 : maxMargin); 1401 }, 1402 1403 // @method toString(): String 1404 // Returns a string representation of the point (for debugging purposes). 1405 toString: function (precision) { 1406 return 'LatLng(' + 1407 formatNum(this.lat, precision) + ', ' + 1408 formatNum(this.lng, precision) + ')'; 1409 }, 1410 1411 // @method distanceTo(otherLatLng: LatLng): Number 1412 // Returns the distance (in meters) to the given `LatLng` calculated using the [Spherical Law of Cosines](https://en.wikipedia.org/wiki/Spherical_law_of_cosines). 1413 distanceTo: function (other) { 1414 return Earth.distance(this, toLatLng(other)); 1415 }, 1416 1417 // @method wrap(): LatLng 1418 // Returns a new `LatLng` object with the longitude wrapped so it's always between -180 and +180 degrees. 1419 wrap: function () { 1420 return Earth.wrapLatLng(this); 1421 }, 1422 1423 // @method toBounds(sizeInMeters: Number): LatLngBounds 1424 // Returns a new `LatLngBounds` object in which each boundary is `sizeInMeters/2` meters apart from the `LatLng`. 1425 toBounds: function (sizeInMeters) { 1426 var latAccuracy = 180 * sizeInMeters / 40075017, 1427 lngAccuracy = latAccuracy / Math.cos((Math.PI / 180) * this.lat); 1428 1429 return toLatLngBounds( 1430 [this.lat - latAccuracy, this.lng - lngAccuracy], 1431 [this.lat + latAccuracy, this.lng + lngAccuracy]); 1432 }, 1433 1434 clone: function () { 1435 return new LatLng(this.lat, this.lng, this.alt); 1436 } 1437 }; 1438 1439 1440 1441 // @factory L.latLng(latitude: Number, longitude: Number, altitude?: Number): LatLng 1442 // Creates an object representing a geographical point with the given latitude and longitude (and optionally altitude). 1443 1444 // @alternative 1445 // @factory L.latLng(coords: Array): LatLng 1446 // Expects an array of the form `[Number, Number]` or `[Number, Number, Number]` instead. 1447 1448 // @alternative 1449 // @factory L.latLng(coords: Object): LatLng 1450 // Expects an plain object of the form `{lat: Number, lng: Number}` or `{lat: Number, lng: Number, alt: Number}` instead. 1451 1452 function toLatLng(a, b, c) { 1453 if (a instanceof LatLng) { 1454 return a; 1455 } 1456 if (isArray(a) && typeof a[0] !== 'object') { 1457 if (a.length === 3) { 1458 return new LatLng(a[0], a[1], a[2]); 1459 } 1460 if (a.length === 2) { 1461 return new LatLng(a[0], a[1]); 1462 } 1463 return null; 1464 } 1465 if (a === undefined || a === null) { 1466 return a; 1467 } 1468 if (typeof a === 'object' && 'lat' in a) { 1469 return new LatLng(a.lat, 'lng' in a ? a.lng : a.lon, a.alt); 1470 } 1471 if (b === undefined) { 1472 return null; 1473 } 1474 return new LatLng(a, b, c); 1475 } 1476 1477 /* 1478 * @namespace CRS 1479 * @crs L.CRS.Base 1480 * Object that defines coordinate reference systems for projecting 1481 * geographical points into pixel (screen) coordinates and back (and to 1482 * coordinates in other units for [WMS](https://en.wikipedia.org/wiki/Web_Map_Service) services). See 1483 * [spatial reference system](http://en.wikipedia.org/wiki/Coordinate_reference_system). 1484 * 1485 * Leaflet defines the most usual CRSs by default. If you want to use a 1486 * CRS not defined by default, take a look at the 1487 * [Proj4Leaflet](https://github.com/kartena/Proj4Leaflet) plugin. 1488 * 1489 * Note that the CRS instances do not inherit from Leafet's `Class` object, 1490 * and can't be instantiated. Also, new classes can't inherit from them, 1491 * and methods can't be added to them with the `include` function. 1492 */ 1493 1494 var CRS = { 1495 // @method latLngToPoint(latlng: LatLng, zoom: Number): Point 1496 // Projects geographical coordinates into pixel coordinates for a given zoom. 1497 latLngToPoint: function (latlng, zoom) { 1498 var projectedPoint = this.projection.project(latlng), 1499 scale = this.scale(zoom); 1500 1501 return this.transformation._transform(projectedPoint, scale); 1502 }, 1503 1504 // @method pointToLatLng(point: Point, zoom: Number): LatLng 1505 // The inverse of `latLngToPoint`. Projects pixel coordinates on a given 1506 // zoom into geographical coordinates. 1507 pointToLatLng: function (point, zoom) { 1508 var scale = this.scale(zoom), 1509 untransformedPoint = this.transformation.untransform(point, scale); 1510 1511 return this.projection.unproject(untransformedPoint); 1512 }, 1513 1514 // @method project(latlng: LatLng): Point 1515 // Projects geographical coordinates into coordinates in units accepted for 1516 // this CRS (e.g. meters for EPSG:3857, for passing it to WMS services). 1517 project: function (latlng) { 1518 return this.projection.project(latlng); 1519 }, 1520 1521 // @method unproject(point: Point): LatLng 1522 // Given a projected coordinate returns the corresponding LatLng. 1523 // The inverse of `project`. 1524 unproject: function (point) { 1525 return this.projection.unproject(point); 1526 }, 1527 1528 // @method scale(zoom: Number): Number 1529 // Returns the scale used when transforming projected coordinates into 1530 // pixel coordinates for a particular zoom. For example, it returns 1531 // `256 * 2^zoom` for Mercator-based CRS. 1532 scale: function (zoom) { 1533 return 256 * Math.pow(2, zoom); 1534 }, 1535 1536 // @method zoom(scale: Number): Number 1537 // Inverse of `scale()`, returns the zoom level corresponding to a scale 1538 // factor of `scale`. 1539 zoom: function (scale) { 1540 return Math.log(scale / 256) / Math.LN2; 1541 }, 1542 1543 // @method getProjectedBounds(zoom: Number): Bounds 1544 // Returns the projection's bounds scaled and transformed for the provided `zoom`. 1545 getProjectedBounds: function (zoom) { 1546 if (this.infinite) { return null; } 1547 1548 var b = this.projection.bounds, 1549 s = this.scale(zoom), 1550 min = this.transformation.transform(b.min, s), 1551 max = this.transformation.transform(b.max, s); 1552 1553 return new Bounds(min, max); 1554 }, 1555 1556 // @method distance(latlng1: LatLng, latlng2: LatLng): Number 1557 // Returns the distance between two geographical coordinates. 1558 1559 // @property code: String 1560 // Standard code name of the CRS passed into WMS services (e.g. `'EPSG:3857'`) 1561 // 1562 // @property wrapLng: Number[] 1563 // An array of two numbers defining whether the longitude (horizontal) coordinate 1564 // axis wraps around a given range and how. Defaults to `[-180, 180]` in most 1565 // geographical CRSs. If `undefined`, the longitude axis does not wrap around. 1566 // 1567 // @property wrapLat: Number[] 1568 // Like `wrapLng`, but for the latitude (vertical) axis. 1569 1570 // wrapLng: [min, max], 1571 // wrapLat: [min, max], 1572 1573 // @property infinite: Boolean 1574 // If true, the coordinate space will be unbounded (infinite in both axes) 1575 infinite: false, 1576 1577 // @method wrapLatLng(latlng: LatLng): LatLng 1578 // Returns a `LatLng` where lat and lng has been wrapped according to the 1579 // CRS's `wrapLat` and `wrapLng` properties, if they are outside the CRS's bounds. 1580 wrapLatLng: function (latlng) { 1581 var lng = this.wrapLng ? wrapNum(latlng.lng, this.wrapLng, true) : latlng.lng, 1582 lat = this.wrapLat ? wrapNum(latlng.lat, this.wrapLat, true) : latlng.lat, 1583 alt = latlng.alt; 1584 1585 return new LatLng(lat, lng, alt); 1586 }, 1587 1588 // @method wrapLatLngBounds(bounds: LatLngBounds): LatLngBounds 1589 // Returns a `LatLngBounds` with the same size as the given one, ensuring 1590 // that its center is within the CRS's bounds. 1591 // Only accepts actual `L.LatLngBounds` instances, not arrays. 1592 wrapLatLngBounds: function (bounds) { 1593 var center = bounds.getCenter(), 1594 newCenter = this.wrapLatLng(center), 1595 latShift = center.lat - newCenter.lat, 1596 lngShift = center.lng - newCenter.lng; 1597 1598 if (latShift === 0 && lngShift === 0) { 1599 return bounds; 1600 } 1601 1602 var sw = bounds.getSouthWest(), 1603 ne = bounds.getNorthEast(), 1604 newSw = new LatLng(sw.lat - latShift, sw.lng - lngShift), 1605 newNe = new LatLng(ne.lat - latShift, ne.lng - lngShift); 1606 1607 return new LatLngBounds(newSw, newNe); 1608 } 1609 }; 1610 1611 /* 1612 * @namespace CRS 1613 * @crs L.CRS.Earth 1614 * 1615 * Serves as the base for CRS that are global such that they cover the earth. 1616 * Can only be used as the base for other CRS and cannot be used directly, 1617 * since it does not have a `code`, `projection` or `transformation`. `distance()` returns 1618 * meters. 1619 */ 1620 1621 var Earth = extend({}, CRS, { 1622 wrapLng: [-180, 180], 1623 1624 // Mean Earth Radius, as recommended for use by 1625 // the International Union of Geodesy and Geophysics, 1626 // see http://rosettacode.org/wiki/Haversine_formula 1627 R: 6371000, 1628 1629 // distance between two geographical points using spherical law of cosines approximation 1630 distance: function (latlng1, latlng2) { 1631 var rad = Math.PI / 180, 1632 lat1 = latlng1.lat * rad, 1633 lat2 = latlng2.lat * rad, 1634 sinDLat = Math.sin((latlng2.lat - latlng1.lat) * rad / 2), 1635 sinDLon = Math.sin((latlng2.lng - latlng1.lng) * rad / 2), 1636 a = sinDLat * sinDLat + Math.cos(lat1) * Math.cos(lat2) * sinDLon * sinDLon, 1637 c = 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a)); 1638 return this.R * c; 1639 } 1640 }); 1641 1642 /* 1643 * @namespace Projection 1644 * @projection L.Projection.SphericalMercator 1645 * 1646 * Spherical Mercator projection — the most common projection for online maps, 1647 * used by almost all free and commercial tile providers. Assumes that Earth is 1648 * a sphere. Used by the `EPSG:3857` CRS. 1649 */ 1650 1651 var earthRadius = 6378137; 1652 1653 var SphericalMercator = { 1654 1655 R: earthRadius, 1656 MAX_LATITUDE: 85.0511287798, 1657 1658 project: function (latlng) { 1659 var d = Math.PI / 180, 1660 max = this.MAX_LATITUDE, 1661 lat = Math.max(Math.min(max, latlng.lat), -max), 1662 sin = Math.sin(lat * d); 1663 1664 return new Point( 1665 this.R * latlng.lng * d, 1666 this.R * Math.log((1 + sin) / (1 - sin)) / 2); 1667 }, 1668 1669 unproject: function (point) { 1670 var d = 180 / Math.PI; 1671 1672 return new LatLng( 1673 (2 * Math.atan(Math.exp(point.y / this.R)) - (Math.PI / 2)) * d, 1674 point.x * d / this.R); 1675 }, 1676 1677 bounds: (function () { 1678 var d = earthRadius * Math.PI; 1679 return new Bounds([-d, -d], [d, d]); 1680 })() 1681 }; 1682 1683 /* 1684 * @class Transformation 1685 * @aka L.Transformation 1686 * 1687 * Represents an affine transformation: a set of coefficients `a`, `b`, `c`, `d` 1688 * for transforming a point of a form `(x, y)` into `(a*x + b, c*y + d)` and doing 1689 * the reverse. Used by Leaflet in its projections code. 1690 * 1691 * @example 1692 * 1693 * ```js 1694 * var transformation = L.transformation(2, 5, -1, 10), 1695 * p = L.point(1, 2), 1696 * p2 = transformation.transform(p), // L.point(7, 8) 1697 * p3 = transformation.untransform(p2); // L.point(1, 2) 1698 * ``` 1699 */ 1700 1701 1702 // factory new L.Transformation(a: Number, b: Number, c: Number, d: Number) 1703 // Creates a `Transformation` object with the given coefficients. 1704 function Transformation(a, b, c, d) { 1705 if (isArray(a)) { 1706 // use array properties 1707 this._a = a[0]; 1708 this._b = a[1]; 1709 this._c = a[2]; 1710 this._d = a[3]; 1711 return; 1712 } 1713 this._a = a; 1714 this._b = b; 1715 this._c = c; 1716 this._d = d; 1717 } 1718 1719 Transformation.prototype = { 1720 // @method transform(point: Point, scale?: Number): Point 1721 // Returns a transformed point, optionally multiplied by the given scale. 1722 // Only accepts actual `L.Point` instances, not arrays. 1723 transform: function (point, scale) { // (Point, Number) -> Point 1724 return this._transform(point.clone(), scale); 1725 }, 1726 1727 // destructive transform (faster) 1728 _transform: function (point, scale) { 1729 scale = scale || 1; 1730 point.x = scale * (this._a * point.x + this._b); 1731 point.y = scale * (this._c * point.y + this._d); 1732 return point; 1733 }, 1734 1735 // @method untransform(point: Point, scale?: Number): Point 1736 // Returns the reverse transformation of the given point, optionally divided 1737 // by the given scale. Only accepts actual `L.Point` instances, not arrays. 1738 untransform: function (point, scale) { 1739 scale = scale || 1; 1740 return new Point( 1741 (point.x / scale - this._b) / this._a, 1742 (point.y / scale - this._d) / this._c); 1743 } 1744 }; 1745 1746 // factory L.transformation(a: Number, b: Number, c: Number, d: Number) 1747 1748 // @factory L.transformation(a: Number, b: Number, c: Number, d: Number) 1749 // Instantiates a Transformation object with the given coefficients. 1750 1751 // @alternative 1752 // @factory L.transformation(coefficients: Array): Transformation 1753 // Expects an coefficients array of the form 1754 // `[a: Number, b: Number, c: Number, d: Number]`. 1755 1756 function toTransformation(a, b, c, d) { 1757 return new Transformation(a, b, c, d); 1758 } 1759 1760 /* 1761 * @namespace CRS 1762 * @crs L.CRS.EPSG3857 1763 * 1764 * The most common CRS for online maps, used by almost all free and commercial 1765 * tile providers. Uses Spherical Mercator projection. Set in by default in 1766 * Map's `crs` option. 1767 */ 1768 1769 var EPSG3857 = extend({}, Earth, { 1770 code: 'EPSG:3857', 1771 projection: SphericalMercator, 1772 1773 transformation: (function () { 1774 var scale = 0.5 / (Math.PI * SphericalMercator.R); 1775 return toTransformation(scale, 0.5, -scale, 0.5); 1776 }()) 1777 }); 1778 1779 var EPSG900913 = extend({}, EPSG3857, { 1780 code: 'EPSG:900913' 1781 }); 1782 1783 // @namespace SVG; @section 1784 // There are several static functions which can be called without instantiating L.SVG: 1785 1786 // @function create(name: String): SVGElement 1787 // Returns a instance of [SVGElement](https://developer.mozilla.org/docs/Web/API/SVGElement), 1788 // corresponding to the class name passed. For example, using 'line' will return 1789 // an instance of [SVGLineElement](https://developer.mozilla.org/docs/Web/API/SVGLineElement). 1790 function svgCreate(name) { 1791 return document.createElementNS('http://www.w3.org/2000/svg', name); 1792 } 1793 1794 // @function pointsToPath(rings: Point[], closed: Boolean): String 1795 // Generates a SVG path string for multiple rings, with each ring turning 1796 // into "M..L..L.." instructions 1797 function pointsToPath(rings, closed) { 1798 var str = '', 1799 i, j, len, len2, points, p; 1800 1801 for (i = 0, len = rings.length; i < len; i++) { 1802 points = rings[i]; 1803 1804 for (j = 0, len2 = points.length; j < len2; j++) { 1805 p = points[j]; 1806 str += (j ? 'L' : 'M') + p.x + ' ' + p.y; 1807 } 1808 1809 // closes the ring for polygons; "x" is VML syntax 1810 str += closed ? (svg ? 'z' : 'x') : ''; 1811 } 1812 1813 // SVG complains about empty path strings 1814 return str || 'M0 0'; 1815 } 1816 1817 /* 1818 * @namespace Browser 1819 * @aka L.Browser 1820 * 1821 * A namespace with static properties for browser/feature detection used by Leaflet internally. 1822 * 1823 * @example 1824 * 1825 * ```js 1826 * if (L.Browser.ielt9) { 1827 * alert('Upgrade your browser, dude!'); 1828 * } 1829 * ``` 1830 */ 1831 1832 var style$1 = document.documentElement.style; 1833 1834 // @property ie: Boolean; `true` for all Internet Explorer versions (not Edge). 1835 var ie = 'ActiveXObject' in window; 1836 1837 // @property ielt9: Boolean; `true` for Internet Explorer versions less than 9. 1838 var ielt9 = ie && !document.addEventListener; 1839 1840 // @property edge: Boolean; `true` for the Edge web browser. 1841 var edge = 'msLaunchUri' in navigator && !('documentMode' in document); 1842 1843 // @property webkit: Boolean; 1844 // `true` for webkit-based browsers like Chrome and Safari (including mobile versions). 1845 var webkit = userAgentContains('webkit'); 1846 1847 // @property android: Boolean 1848 // `true` for any browser running on an Android platform. 1849 var android = userAgentContains('android'); 1850 1851 // @property android23: Boolean; `true` for browsers running on Android 2 or Android 3. 1852 var android23 = userAgentContains('android 2') || userAgentContains('android 3'); 1853 1854 /* See https://stackoverflow.com/a/17961266 for details on detecting stock Android */ 1855 var webkitVer = parseInt(/WebKit\/([0-9]+)|$/.exec(navigator.userAgent)[1], 10); // also matches AppleWebKit 1856 // @property androidStock: Boolean; `true` for the Android stock browser (i.e. not Chrome) 1857 var androidStock = android && userAgentContains('Google') && webkitVer < 537 && !('AudioNode' in window); 1858 1859 // @property opera: Boolean; `true` for the Opera browser 1860 var opera = !!window.opera; 1861 1862 // @property chrome: Boolean; `true` for the Chrome browser. 1863 var chrome = userAgentContains('chrome'); 1864 1865 // @property gecko: Boolean; `true` for gecko-based browsers like Firefox. 1866 var gecko = userAgentContains('gecko') && !webkit && !opera && !ie; 1867 1868 // @property safari: Boolean; `true` for the Safari browser. 1869 var safari = !chrome && userAgentContains('safari'); 1870 1871 var phantom = userAgentContains('phantom'); 1872 1873 // @property opera12: Boolean 1874 // `true` for the Opera browser supporting CSS transforms (version 12 or later). 1875 var opera12 = 'OTransition' in style$1; 1876 1877 // @property win: Boolean; `true` when the browser is running in a Windows platform 1878 var win = navigator.platform.indexOf('Win') === 0; 1879 1880 // @property ie3d: Boolean; `true` for all Internet Explorer versions supporting CSS transforms. 1881 var ie3d = ie && ('transition' in style$1); 1882 1883 // @property webkit3d: Boolean; `true` for webkit-based browsers supporting CSS transforms. 1884 var webkit3d = ('WebKitCSSMatrix' in window) && ('m11' in new window.WebKitCSSMatrix()) && !android23; 1885 1886 // @property gecko3d: Boolean; `true` for gecko-based browsers supporting CSS transforms. 1887 var gecko3d = 'MozPerspective' in style$1; 1888 1889 // @property any3d: Boolean 1890 // `true` for all browsers supporting CSS transforms. 1891 var any3d = !window.L_DISABLE_3D && (ie3d || webkit3d || gecko3d) && !opera12 && !phantom; 1892 1893 // @property mobile: Boolean; `true` for all browsers running in a mobile device. 1894 var mobile = typeof orientation !== 'undefined' || userAgentContains('mobile'); 1895 1896 // @property mobileWebkit: Boolean; `true` for all webkit-based browsers in a mobile device. 1897 var mobileWebkit = mobile && webkit; 1898 1899 // @property mobileWebkit3d: Boolean 1900 // `true` for all webkit-based browsers in a mobile device supporting CSS transforms. 1901 var mobileWebkit3d = mobile && webkit3d; 1902 1903 // @property msPointer: Boolean 1904 // `true` for browsers implementing the Microsoft touch events model (notably IE10). 1905 var msPointer = !window.PointerEvent && window.MSPointerEvent; 1906 1907 // @property pointer: Boolean 1908 // `true` for all browsers supporting [pointer events](https://msdn.microsoft.com/en-us/library/dn433244%28v=vs.85%29.aspx). 1909 var pointer = !webkit && !!(window.PointerEvent || msPointer); 1910 1911 // @property touch: Boolean 1912 // `true` for all browsers supporting [touch events](https://developer.mozilla.org/docs/Web/API/Touch_events). 1913 // This does not necessarily mean that the browser is running in a computer with 1914 // a touchscreen, it only means that the browser is capable of understanding 1915 // touch events. 1916 var touch = !window.L_NO_TOUCH && (pointer || 'ontouchstart' in window || 1917 (window.DocumentTouch && document instanceof window.DocumentTouch)); 1918 1919 // @property mobileOpera: Boolean; `true` for the Opera browser in a mobile device. 1920 var mobileOpera = mobile && opera; 1921 1922 // @property mobileGecko: Boolean 1923 // `true` for gecko-based browsers running in a mobile device. 1924 var mobileGecko = mobile && gecko; 1925 1926 // @property retina: Boolean 1927 // `true` for browsers on a high-resolution "retina" screen or on any screen when browser's display zoom is more than 100%. 1928 var retina = (window.devicePixelRatio || (window.screen.deviceXDPI / window.screen.logicalXDPI)) > 1; 1929 1930 // @property passiveEvents: Boolean 1931 // `true` for browsers that support passive events. 1932 var passiveEvents = (function () { 1933 var supportsPassiveOption = false; 1934 try { 1935 var opts = Object.defineProperty({}, 'passive', { 1936 get: function () { 1937 supportsPassiveOption = true; 1938 } 1939 }); 1940 window.addEventListener('testPassiveEventSupport', falseFn, opts); 1941 window.removeEventListener('testPassiveEventSupport', falseFn, opts); 1942 } catch (e) { 1943 // Errors can safely be ignored since this is only a browser support test. 1944 } 1945 return supportsPassiveOption; 1946 }); 1947 1948 // @property canvas: Boolean 1949 // `true` when the browser supports [`<canvas>`](https://developer.mozilla.org/docs/Web/API/Canvas_API). 1950 var canvas = (function () { 1951 return !!document.createElement('canvas').getContext; 1952 }()); 1953 1954 // @property svg: Boolean 1955 // `true` when the browser supports [SVG](https://developer.mozilla.org/docs/Web/SVG). 1956 var svg = !!(document.createElementNS && svgCreate('svg').createSVGRect); 1957 1958 // @property vml: Boolean 1959 // `true` if the browser supports [VML](https://en.wikipedia.org/wiki/Vector_Markup_Language). 1960 var vml = !svg && (function () { 1961 try { 1962 var div = document.createElement('div'); 1963 div.innerHTML = '<v:shape adj="1"/>'; 1964 1965 var shape = div.firstChild; 1966 shape.style.behavior = 'url(#default#VML)'; 1967 1968 return shape && (typeof shape.adj === 'object'); 1969 1970 } catch (e) { 1971 return false; 1972 } 1973 }()); 1974 1975 1976 function userAgentContains(str) { 1977 return navigator.userAgent.toLowerCase().indexOf(str) >= 0; 1978 } 1979 1980 1981 var Browser = (Object.freeze || Object)({ 1982 ie: ie, 1983 ielt9: ielt9, 1984 edge: edge, 1985 webkit: webkit, 1986 android: android, 1987 android23: android23, 1988 androidStock: androidStock, 1989 opera: opera, 1990 chrome: chrome, 1991 gecko: gecko, 1992 safari: safari, 1993 phantom: phantom, 1994 opera12: opera12, 1995 win: win, 1996 ie3d: ie3d, 1997 webkit3d: webkit3d, 1998 gecko3d: gecko3d, 1999 any3d: any3d, 2000 mobile: mobile, 2001 mobileWebkit: mobileWebkit, 2002 mobileWebkit3d: mobileWebkit3d, 2003 msPointer: msPointer, 2004 pointer: pointer, 2005 touch: touch, 2006 mobileOpera: mobileOpera, 2007 mobileGecko: mobileGecko, 2008 retina: retina, 2009 passiveEvents: passiveEvents, 2010 canvas: canvas, 2011 svg: svg, 2012 vml: vml 2013 }); 2014 2015 /* 2016 * Extends L.DomEvent to provide touch support for Internet Explorer and Windows-based devices. 2017 */ 2018 2019 2020 var POINTER_DOWN = msPointer ? 'MSPointerDown' : 'pointerdown'; 2021 var POINTER_MOVE = msPointer ? 'MSPointerMove' : 'pointermove'; 2022 var POINTER_UP = msPointer ? 'MSPointerUp' : 'pointerup'; 2023 var POINTER_CANCEL = msPointer ? 'MSPointerCancel' : 'pointercancel'; 2024 var TAG_WHITE_LIST = ['INPUT', 'SELECT', 'OPTION']; 2025 2026 var _pointers = {}; 2027 var _pointerDocListener = false; 2028 2029 // DomEvent.DoubleTap needs to know about this 2030 var _pointersCount = 0; 2031 2032 // Provides a touch events wrapper for (ms)pointer events. 2033 // ref http://www.w3.org/TR/pointerevents/ https://www.w3.org/Bugs/Public/show_bug.cgi?id=22890 2034 2035 function addPointerListener(obj, type, handler, id) { 2036 if (type === 'touchstart') { 2037 _addPointerStart(obj, handler, id); 2038 2039 } else if (type === 'touchmove') { 2040 _addPointerMove(obj, handler, id); 2041 2042 } else if (type === 'touchend') { 2043 _addPointerEnd(obj, handler, id); 2044 } 2045 2046 return this; 2047 } 2048 2049 function removePointerListener(obj, type, id) { 2050 var handler = obj['_leaflet_' + type + id]; 2051 2052 if (type === 'touchstart') { 2053 obj.removeEventListener(POINTER_DOWN, handler, false); 2054 2055 } else if (type === 'touchmove') { 2056 obj.removeEventListener(POINTER_MOVE, handler, false); 2057 2058 } else if (type === 'touchend') { 2059 obj.removeEventListener(POINTER_UP, handler, false); 2060 obj.removeEventListener(POINTER_CANCEL, handler, false); 2061 } 2062 2063 return this; 2064 } 2065 2066 function _addPointerStart(obj, handler, id) { 2067 var onDown = bind(function (e) { 2068 if (e.pointerType !== 'mouse' && e.MSPOINTER_TYPE_MOUSE && e.pointerType !== e.MSPOINTER_TYPE_MOUSE) { 2069 // In IE11, some touch events needs to fire for form controls, or 2070 // the controls will stop working. We keep a whitelist of tag names that 2071 // need these events. For other target tags, we prevent default on the event. 2072 if (TAG_WHITE_LIST.indexOf(e.target.tagName) < 0) { 2073 preventDefault(e); 2074 } else { 2075 return; 2076 } 2077 } 2078 2079 _handlePointer(e, handler); 2080 }); 2081 2082 obj['_leaflet_touchstart' + id] = onDown; 2083 obj.addEventListener(POINTER_DOWN, onDown, false); 2084 2085 // need to keep track of what pointers and how many are active to provide e.touches emulation 2086 if (!_pointerDocListener) { 2087 // we listen documentElement as any drags that end by moving the touch off the screen get fired there 2088 document.documentElement.addEventListener(POINTER_DOWN, _globalPointerDown, true); 2089 document.documentElement.addEventListener(POINTER_MOVE, _globalPointerMove, true); 2090 document.documentElement.addEventListener(POINTER_UP, _globalPointerUp, true); 2091 document.documentElement.addEventListener(POINTER_CANCEL, _globalPointerUp, true); 2092 2093 _pointerDocListener = true; 2094 } 2095 } 2096 2097 function _globalPointerDown(e) { 2098 _pointers[e.pointerId] = e; 2099 _pointersCount++; 2100 } 2101 2102 function _globalPointerMove(e) { 2103 if (_pointers[e.pointerId]) { 2104 _pointers[e.pointerId] = e; 2105 } 2106 } 2107 2108 function _globalPointerUp(e) { 2109 delete _pointers[e.pointerId]; 2110 _pointersCount--; 2111 } 2112 2113 function _handlePointer(e, handler) { 2114 e.touches = []; 2115 for (var i in _pointers) { 2116 e.touches.push(_pointers[i]); 2117 } 2118 e.changedTouches = [e]; 2119 2120 handler(e); 2121 } 2122 2123 function _addPointerMove(obj, handler, id) { 2124 var onMove = function (e) { 2125 // don't fire touch moves when mouse isn't down 2126 if ((e.pointerType === e.MSPOINTER_TYPE_MOUSE || e.pointerType === 'mouse') && e.buttons === 0) { return; } 2127 2128 _handlePointer(e, handler); 2129 }; 2130 2131 obj['_leaflet_touchmove' + id] = onMove; 2132 obj.addEventListener(POINTER_MOVE, onMove, false); 2133 } 2134 2135 function _addPointerEnd(obj, handler, id) { 2136 var onUp = function (e) { 2137 _handlePointer(e, handler); 2138 }; 2139 2140 obj['_leaflet_touchend' + id] = onUp; 2141 obj.addEventListener(POINTER_UP, onUp, false); 2142 obj.addEventListener(POINTER_CANCEL, onUp, false); 2143 } 2144 2145 /* 2146 * Extends the event handling code with double tap support for mobile browsers. 2147 */ 2148 2149 var _touchstart = msPointer ? 'MSPointerDown' : pointer ? 'pointerdown' : 'touchstart'; 2150 var _touchend = msPointer ? 'MSPointerUp' : pointer ? 'pointerup' : 'touchend'; 2151 var _pre = '_leaflet_'; 2152 2153 // inspired by Zepto touch code by Thomas Fuchs 2154 function addDoubleTapListener(obj, handler, id) { 2155 var last, touch$$1, 2156 doubleTap = false, 2157 delay = 250; 2158 2159 function onTouchStart(e) { 2160 var count; 2161 2162 if (pointer) { 2163 if ((!edge) || e.pointerType === 'mouse') { return; } 2164 count = _pointersCount; 2165 } else { 2166 count = e.touches.length; 2167 } 2168 2169 if (count > 1) { return; } 2170 2171 var now = Date.now(), 2172 delta = now - (last || now); 2173 2174 touch$$1 = e.touches ? e.touches[0] : e; 2175 doubleTap = (delta > 0 && delta <= delay); 2176 last = now; 2177 } 2178 2179 function onTouchEnd(e) { 2180 if (doubleTap && !touch$$1.cancelBubble) { 2181 if (pointer) { 2182 if ((!edge) || e.pointerType === 'mouse') { return; } 2183 // work around .type being readonly with MSPointer* events 2184 var newTouch = {}, 2185 prop, i; 2186 2187 for (i in touch$$1) { 2188 prop = touch$$1[i]; 2189 newTouch[i] = prop && prop.bind ? prop.bind(touch$$1) : prop; 2190 } 2191 touch$$1 = newTouch; 2192 } 2193 touch$$1.type = 'dblclick'; 2194 touch$$1.button = 0; 2195 handler(touch$$1); 2196 last = null; 2197 } 2198 } 2199 2200 obj[_pre + _touchstart + id] = onTouchStart; 2201 obj[_pre + _touchend + id] = onTouchEnd; 2202 obj[_pre + 'dblclick' + id] = handler; 2203 2204 obj.addEventListener(_touchstart, onTouchStart, passiveEvents ? {passive: false} : false); 2205 obj.addEventListener(_touchend, onTouchEnd, passiveEvents ? {passive: false} : false); 2206 2207 // On some platforms (notably, chrome<55 on win10 + touchscreen + mouse), 2208 // the browser doesn't fire touchend/pointerup events but does fire 2209 // native dblclicks. See #4127. 2210 // Edge 14 also fires native dblclicks, but only for pointerType mouse, see #5180. 2211 obj.addEventListener('dblclick', handler, false); 2212 2213 return this; 2214 } 2215 2216 function removeDoubleTapListener(obj, id) { 2217 var touchstart = obj[_pre + _touchstart + id], 2218 touchend = obj[_pre + _touchend + id], 2219 dblclick = obj[_pre + 'dblclick' + id]; 2220 2221 obj.removeEventListener(_touchstart, touchstart, passiveEvents ? {passive: false} : false); 2222 obj.removeEventListener(_touchend, touchend, passiveEvents ? {passive: false} : false); 2223 if (!edge) { 2224 obj.removeEventListener('dblclick', dblclick, false); 2225 } 2226 2227 return this; 2228 } 2229 2230 /* 2231 * @namespace DomUtil 2232 * 2233 * Utility functions to work with the [DOM](https://developer.mozilla.org/docs/Web/API/Document_Object_Model) 2234 * tree, used by Leaflet internally. 2235 * 2236 * Most functions expecting or returning a `HTMLElement` also work for 2237 * SVG elements. The only difference is that classes refer to CSS classes 2238 * in HTML and SVG classes in SVG. 2239 */ 2240 2241 2242 // @property TRANSFORM: String 2243 // Vendor-prefixed transform style name (e.g. `'webkitTransform'` for WebKit). 2244 var TRANSFORM = testProp( 2245 ['transform', 'webkitTransform', 'OTransform', 'MozTransform', 'msTransform']); 2246 2247 // webkitTransition comes first because some browser versions that drop vendor prefix don't do 2248 // the same for the transitionend event, in particular the Android 4.1 stock browser 2249 2250 // @property TRANSITION: String 2251 // Vendor-prefixed transition style name. 2252 var TRANSITION = testProp( 2253 ['webkitTransition', 'transition', 'OTransition', 'MozTransition', 'msTransition']); 2254 2255 // @property TRANSITION_END: String 2256 // Vendor-prefixed transitionend event name. 2257 var TRANSITION_END = 2258 TRANSITION === 'webkitTransition' || TRANSITION === 'OTransition' ? TRANSITION + 'End' : 'transitionend'; 2259 2260 2261 // @function get(id: String|HTMLElement): HTMLElement 2262 // Returns an element given its DOM id, or returns the element itself 2263 // if it was passed directly. 2264 function get(id) { 2265 return typeof id === 'string' ? document.getElementById(id) : id; 2266 } 2267 2268 // @function getStyle(el: HTMLElement, styleAttrib: String): String 2269 // Returns the value for a certain style attribute on an element, 2270 // including computed values or values set through CSS. 2271 function getStyle(el, style) { 2272 var value = el.style[style] || (el.currentStyle && el.currentStyle[style]); 2273 2274 if ((!value || value === 'auto') && document.defaultView) { 2275 var css = document.defaultView.getComputedStyle(el, null); 2276 value = css ? css[style] : null; 2277 } 2278 return value === 'auto' ? null : value; 2279 } 2280 2281 // @function create(tagName: String, className?: String, container?: HTMLElement): HTMLElement 2282 // Creates an HTML element with `tagName`, sets its class to `className`, and optionally appends it to `container` element. 2283 function create$1(tagName, className, container) { 2284 var el = document.createElement(tagName); 2285 el.className = className || ''; 2286 2287 if (container) { 2288 container.appendChild(el); 2289 } 2290 return el; 2291 } 2292 2293 // @function remove(el: HTMLElement) 2294 // Removes `el` from its parent element 2295 function remove(el) { 2296 var parent = el.parentNode; 2297 if (parent) { 2298 parent.removeChild(el); 2299 } 2300 } 2301 2302 // @function empty(el: HTMLElement) 2303 // Removes all of `el`'s children elements from `el` 2304 function empty(el) { 2305 while (el.firstChild) { 2306 el.removeChild(el.firstChild); 2307 } 2308 } 2309 2310 // @function toFront(el: HTMLElement) 2311 // Makes `el` the last child of its parent, so it renders in front of the other children. 2312 function toFront(el) { 2313 var parent = el.parentNode; 2314 if (parent && parent.lastChild !== el) { 2315 parent.appendChild(el); 2316 } 2317 } 2318 2319 // @function toBack(el: HTMLElement) 2320 // Makes `el` the first child of its parent, so it renders behind the other children. 2321 function toBack(el) { 2322 var parent = el.parentNode; 2323 if (parent && parent.firstChild !== el) { 2324 parent.insertBefore(el, parent.firstChild); 2325 } 2326 } 2327 2328 // @function hasClass(el: HTMLElement, name: String): Boolean 2329 // Returns `true` if the element's class attribute contains `name`. 2330 function hasClass(el, name) { 2331 if (el.classList !== undefined) { 2332 return el.classList.contains(name); 2333 } 2334 var className = getClass(el); 2335 return className.length > 0 && new RegExp('(^|\\s)' + name + '(\\s|$)').test(className); 2336 } 2337 2338 // @function addClass(el: HTMLElement, name: String) 2339 // Adds `name` to the element's class attribute. 2340 function addClass(el, name) { 2341 if (el.classList !== undefined) { 2342 var classes = splitWords(name); 2343 for (var i = 0, len = classes.length; i < len; i++) { 2344 el.classList.add(classes[i]); 2345 } 2346 } else if (!hasClass(el, name)) { 2347 var className = getClass(el); 2348 setClass(el, (className ? className + ' ' : '') + name); 2349 } 2350 } 2351 2352 // @function removeClass(el: HTMLElement, name: String) 2353 // Removes `name` from the element's class attribute. 2354 function removeClass(el, name) { 2355 if (el.classList !== undefined) { 2356 el.classList.remove(name); 2357 } else { 2358 setClass(el, trim((' ' + getClass(el) + ' ').replace(' ' + name + ' ', ' '))); 2359 } 2360 } 2361 2362 // @function setClass(el: HTMLElement, name: String) 2363 // Sets the element's class. 2364 function setClass(el, name) { 2365 if (el.className.baseVal === undefined) { 2366 el.className = name; 2367 } else { 2368 // in case of SVG element 2369 el.className.baseVal = name; 2370 } 2371 } 2372 2373 // @function getClass(el: HTMLElement): String 2374 // Returns the element's class. 2375 function getClass(el) { 2376 // Check if the element is an SVGElementInstance and use the correspondingElement instead 2377 // (Required for linked SVG elements in IE11.) 2378 if (el.correspondingElement) { 2379 el = el.correspondingElement; 2380 } 2381 return el.className.baseVal === undefined ? el.className : el.className.baseVal; 2382 } 2383 2384 // @function setOpacity(el: HTMLElement, opacity: Number) 2385 // Set the opacity of an element (including old IE support). 2386 // `opacity` must be a number from `0` to `1`. 2387 function setOpacity(el, value) { 2388 if ('opacity' in el.style) { 2389 el.style.opacity = value; 2390 } else if ('filter' in el.style) { 2391 _setOpacityIE(el, value); 2392 } 2393 } 2394 2395 function _setOpacityIE(el, value) { 2396 var filter = false, 2397 filterName = 'DXImageTransform.Microsoft.Alpha'; 2398 2399 // filters collection throws an error if we try to retrieve a filter that doesn't exist 2400 try { 2401 filter = el.filters.item(filterName); 2402 } catch (e) { 2403 // don't set opacity to 1 if we haven't already set an opacity, 2404 // it isn't needed and breaks transparent pngs. 2405 if (value === 1) { return; } 2406 } 2407 2408 value = Math.round(value * 100); 2409 2410 if (filter) { 2411 filter.Enabled = (value !== 100); 2412 filter.Opacity = value; 2413 } else { 2414 el.style.filter += ' progid:' + filterName + '(opacity=' + value + ')'; 2415 } 2416 } 2417 2418 // @function testProp(props: String[]): String|false 2419 // Goes through the array of style names and returns the first name 2420 // that is a valid style name for an element. If no such name is found, 2421 // it returns false. Useful for vendor-prefixed styles like `transform`. 2422 function testProp(props) { 2423 var style = document.documentElement.style; 2424 2425 for (var i = 0; i < props.length; i++) { 2426 if (props[i] in style) { 2427 return props[i]; 2428 } 2429 } 2430 return false; 2431 } 2432 2433 // @function setTransform(el: HTMLElement, offset: Point, scale?: Number) 2434 // Resets the 3D CSS transform of `el` so it is translated by `offset` pixels 2435 // and optionally scaled by `scale`. Does not have an effect if the 2436 // browser doesn't support 3D CSS transforms. 2437 function setTransform(el, offset, scale) { 2438 var pos = offset || new Point(0, 0); 2439 2440 el.style[TRANSFORM] = 2441 (ie3d ? 2442 'translate(' + pos.x + 'px,' + pos.y + 'px)' : 2443 'translate3d(' + pos.x + 'px,' + pos.y + 'px,0)') + 2444 (scale ? ' scale(' + scale + ')' : ''); 2445 } 2446 2447 // @function setPosition(el: HTMLElement, position: Point) 2448 // Sets the position of `el` to coordinates specified by `position`, 2449 // using CSS translate or top/left positioning depending on the browser 2450 // (used by Leaflet internally to position its layers). 2451 function setPosition(el, point) { 2452 2453 /*eslint-disable */ 2454 el._leaflet_pos = point; 2455 /* eslint-enable */ 2456 2457 if (any3d) { 2458 setTransform(el, point); 2459 } else { 2460 el.style.left = point.x + 'px'; 2461 el.style.top = point.y + 'px'; 2462 } 2463 } 2464 2465 // @function getPosition(el: HTMLElement): Point 2466 // Returns the coordinates of an element previously positioned with setPosition. 2467 function getPosition(el) { 2468 // this method is only used for elements previously positioned using setPosition, 2469 // so it's safe to cache the position for performance 2470 2471 return el._leaflet_pos || new Point(0, 0); 2472 } 2473 2474 // @function disableTextSelection() 2475 // Prevents the user from generating `selectstart` DOM events, usually generated 2476 // when the user drags the mouse through a page with text. Used internally 2477 // by Leaflet to override the behaviour of any click-and-drag interaction on 2478 // the map. Affects drag interactions on the whole document. 2479 2480 // @function enableTextSelection() 2481 // Cancels the effects of a previous [`L.DomUtil.disableTextSelection`](#domutil-disabletextselection). 2482 var disableTextSelection; 2483 var enableTextSelection; 2484 var _userSelect; 2485 if ('onselectstart' in document) { 2486 disableTextSelection = function () { 2487 on(window, 'selectstart', preventDefault); 2488 }; 2489 enableTextSelection = function () { 2490 off(window, 'selectstart', preventDefault); 2491 }; 2492 } else { 2493 var userSelectProperty = testProp( 2494 ['userSelect', 'WebkitUserSelect', 'OUserSelect', 'MozUserSelect', 'msUserSelect']); 2495 2496 disableTextSelection = function () { 2497 if (userSelectProperty) { 2498 var style = document.documentElement.style; 2499 _userSelect = style[userSelectProperty]; 2500 style[userSelectProperty] = 'none'; 2501 } 2502 }; 2503 enableTextSelection = function () { 2504 if (userSelectProperty) { 2505 document.documentElement.style[userSelectProperty] = _userSelect; 2506 _userSelect = undefined; 2507 } 2508 }; 2509 } 2510 2511 // @function disableImageDrag() 2512 // As [`L.DomUtil.disableTextSelection`](#domutil-disabletextselection), but 2513 // for `dragstart` DOM events, usually generated when the user drags an image. 2514 function disableImageDrag() { 2515 on(window, 'dragstart', preventDefault); 2516 } 2517 2518 // @function enableImageDrag() 2519 // Cancels the effects of a previous [`L.DomUtil.disableImageDrag`](#domutil-disabletextselection). 2520 function enableImageDrag() { 2521 off(window, 'dragstart', preventDefault); 2522 } 2523 2524 var _outlineElement; 2525 var _outlineStyle; 2526 // @function preventOutline(el: HTMLElement) 2527 // Makes the [outline](https://developer.mozilla.org/docs/Web/CSS/outline) 2528 // of the element `el` invisible. Used internally by Leaflet to prevent 2529 // focusable elements from displaying an outline when the user performs a 2530 // drag interaction on them. 2531 function preventOutline(element) { 2532 while (element.tabIndex === -1) { 2533 element = element.parentNode; 2534 } 2535 if (!element.style) { return; } 2536 restoreOutline(); 2537 _outlineElement = element; 2538 _outlineStyle = element.style.outline; 2539 element.style.outline = 'none'; 2540 on(window, 'keydown', restoreOutline); 2541 } 2542 2543 // @function restoreOutline() 2544 // Cancels the effects of a previous [`L.DomUtil.preventOutline`](). 2545 function restoreOutline() { 2546 if (!_outlineElement) { return; } 2547 _outlineElement.style.outline = _outlineStyle; 2548 _outlineElement = undefined; 2549 _outlineStyle = undefined; 2550 off(window, 'keydown', restoreOutline); 2551 } 2552 2553 // @function getSizedParentNode(el: HTMLElement): HTMLElement 2554 // Finds the closest parent node which size (width and height) is not null. 2555 function getSizedParentNode(element) { 2556 do { 2557 element = element.parentNode; 2558 } while ((!element.offsetWidth || !element.offsetHeight) && element !== document.body); 2559 return element; 2560 } 2561 2562 // @function getScale(el: HTMLElement): Object 2563 // Computes the CSS scale currently applied on the element. 2564 // Returns an object with `x` and `y` members as horizontal and vertical scales respectively, 2565 // and `boundingClientRect` as the result of [`getBoundingClientRect()`](https://developer.mozilla.org/en-US/docs/Web/API/Element/getBoundingClientRect). 2566 function getScale(element) { 2567 var rect = element.getBoundingClientRect(); // Read-only in old browsers. 2568 2569 return { 2570 x: rect.width / element.offsetWidth || 1, 2571 y: rect.height / element.offsetHeight || 1, 2572 boundingClientRect: rect 2573 }; 2574 } 2575 2576 2577 var DomUtil = (Object.freeze || Object)({ 2578 TRANSFORM: TRANSFORM, 2579 TRANSITION: TRANSITION, 2580 TRANSITION_END: TRANSITION_END, 2581 get: get, 2582 getStyle: getStyle, 2583 create: create$1, 2584 remove: remove, 2585 empty: empty, 2586 toFront: toFront, 2587 toBack: toBack, 2588 hasClass: hasClass, 2589 addClass: addClass, 2590 removeClass: removeClass, 2591 setClass: setClass, 2592 getClass: getClass, 2593 setOpacity: setOpacity, 2594 testProp: testProp, 2595 setTransform: setTransform, 2596 setPosition: setPosition, 2597 getPosition: getPosition, 2598 disableTextSelection: disableTextSelection, 2599 enableTextSelection: enableTextSelection, 2600 disableImageDrag: disableImageDrag, 2601 enableImageDrag: enableImageDrag, 2602 preventOutline: preventOutline, 2603 restoreOutline: restoreOutline, 2604 getSizedParentNode: getSizedParentNode, 2605 getScale: getScale 2606 }); 2607 2608 /* 2609 * @namespace DomEvent 2610 * Utility functions to work with the [DOM events](https://developer.mozilla.org/docs/Web/API/Event), used by Leaflet internally. 2611 */ 2612 2613 // Inspired by John Resig, Dean Edwards and YUI addEvent implementations. 2614 2615 // @function on(el: HTMLElement, types: String, fn: Function, context?: Object): this 2616 // Adds a listener function (`fn`) to a particular DOM event type of the 2617 // element `el`. You can optionally specify the context of the listener 2618 // (object the `this` keyword will point to). You can also pass several 2619 // space-separated types (e.g. `'click dblclick'`). 2620 2621 // @alternative 2622 // @function on(el: HTMLElement, eventMap: Object, context?: Object): this 2623 // Adds a set of type/listener pairs, e.g. `{click: onClick, mousemove: onMouseMove}` 2624 function on(obj, types, fn, context) { 2625 2626 if (typeof types === 'object') { 2627 for (var type in types) { 2628 addOne(obj, type, types[type], fn); 2629 } 2630 } else { 2631 types = splitWords(types); 2632 2633 for (var i = 0, len = types.length; i < len; i++) { 2634 addOne(obj, types[i], fn, context); 2635 } 2636 } 2637 2638 return this; 2639 } 2640 2641 var eventsKey = '_leaflet_events'; 2642 2643 // @function off(el: HTMLElement, types: String, fn: Function, context?: Object): this 2644 // Removes a previously added listener function. 2645 // Note that if you passed a custom context to on, you must pass the same 2646 // context to `off` in order to remove the listener. 2647 2648 // @alternative 2649 // @function off(el: HTMLElement, eventMap: Object, context?: Object): this 2650 // Removes a set of type/listener pairs, e.g. `{click: onClick, mousemove: onMouseMove}` 2651 function off(obj, types, fn, context) { 2652 2653 if (typeof types === 'object') { 2654 for (var type in types) { 2655 removeOne(obj, type, types[type], fn); 2656 } 2657 } else if (types) { 2658 types = splitWords(types); 2659 2660 for (var i = 0, len = types.length; i < len; i++) { 2661 removeOne(obj, types[i], fn, context); 2662 } 2663 } else { 2664 for (var j in obj[eventsKey]) { 2665 removeOne(obj, j, obj[eventsKey][j]); 2666 } 2667 delete obj[eventsKey]; 2668 } 2669 2670 return this; 2671 } 2672 2673 function addOne(obj, type, fn, context) { 2674 var id = type + stamp(fn) + (context ? '_' + stamp(context) : ''); 2675 2676 if (obj[eventsKey] && obj[eventsKey][id]) { return this; } 2677 2678 var handler = function (e) { 2679 return fn.call(context || obj, e || window.event); 2680 }; 2681 2682 var originalHandler = handler; 2683 2684 if (pointer && type.indexOf('touch') === 0) { 2685 // Needs DomEvent.Pointer.js 2686 addPointerListener(obj, type, handler, id); 2687 2688 } else if (touch && (type === 'dblclick') && addDoubleTapListener && 2689 !(pointer && chrome)) { 2690 // Chrome >55 does not need the synthetic dblclicks from addDoubleTapListener 2691 // See #5180 2692 addDoubleTapListener(obj, handler, id); 2693 2694 } else if ('addEventListener' in obj) { 2695 2696 if (type === 'mousewheel') { 2697 obj.addEventListener('onwheel' in obj ? 'wheel' : 'mousewheel', handler, passiveEvents ? {passive: false} : false); 2698 2699 } else if ((type === 'mouseenter') || (type === 'mouseleave')) { 2700 handler = function (e) { 2701 e = e || window.event; 2702 if (isExternalTarget(obj, e)) { 2703 originalHandler(e); 2704 } 2705 }; 2706 obj.addEventListener(type === 'mouseenter' ? 'mouseover' : 'mouseout', handler, false); 2707 2708 } else { 2709 if (type === 'click' && android) { 2710 handler = function (e) { 2711 filterClick(e, originalHandler); 2712 }; 2713 } 2714 obj.addEventListener(type, handler, false); 2715 } 2716 2717 } else if ('attachEvent' in obj) { 2718 obj.attachEvent('on' + type, handler); 2719 } 2720 2721 obj[eventsKey] = obj[eventsKey] || {}; 2722 obj[eventsKey][id] = handler; 2723 } 2724 2725 function removeOne(obj, type, fn, context) { 2726 2727 var id = type + stamp(fn) + (context ? '_' + stamp(context) : ''), 2728 handler = obj[eventsKey] && obj[eventsKey][id]; 2729 2730 if (!handler) { return this; } 2731 2732 if (pointer && type.indexOf('touch') === 0) { 2733 removePointerListener(obj, type, id); 2734 2735 } else if (touch && (type === 'dblclick') && removeDoubleTapListener && 2736 !(pointer && chrome)) { 2737 removeDoubleTapListener(obj, id); 2738 2739 } else if ('removeEventListener' in obj) { 2740 2741 if (type === 'mousewheel') { 2742 obj.removeEventListener('onwheel' in obj ? 'wheel' : 'mousewheel', handler, passiveEvents ? {passive: false} : false); 2743 2744 } else { 2745 obj.removeEventListener( 2746 type === 'mouseenter' ? 'mouseover' : 2747 type === 'mouseleave' ? 'mouseout' : type, handler, false); 2748 } 2749 2750 } else if ('detachEvent' in obj) { 2751 obj.detachEvent('on' + type, handler); 2752 } 2753 2754 obj[eventsKey][id] = null; 2755 } 2756 2757 // @function stopPropagation(ev: DOMEvent): this 2758 // Stop the given event from propagation to parent elements. Used inside the listener functions: 2759 // ```js 2760 // L.DomEvent.on(div, 'click', function (ev) { 2761 // L.DomEvent.stopPropagation(ev); 2762 // }); 2763 // ``` 2764 function stopPropagation(e) { 2765 2766 if (e.stopPropagation) { 2767 e.stopPropagation(); 2768 } else if (e.originalEvent) { // In case of Leaflet event. 2769 e.originalEvent._stopped = true; 2770 } else { 2771 e.cancelBubble = true; 2772 } 2773 skipped(e); 2774 2775 return this; 2776 } 2777 2778 // @function disableScrollPropagation(el: HTMLElement): this 2779 // Adds `stopPropagation` to the element's `'mousewheel'` events (plus browser variants). 2780 function disableScrollPropagation(el) { 2781 addOne(el, 'mousewheel', stopPropagation); 2782 return this; 2783 } 2784 2785 // @function disableClickPropagation(el: HTMLElement): this 2786 // Adds `stopPropagation` to the element's `'click'`, `'doubleclick'`, 2787 // `'mousedown'` and `'touchstart'` events (plus browser variants). 2788 function disableClickPropagation(el) { 2789 on(el, 'mousedown touchstart dblclick', stopPropagation); 2790 addOne(el, 'click', fakeStop); 2791 return this; 2792 } 2793 2794 // @function preventDefault(ev: DOMEvent): this 2795 // Prevents the default action of the DOM Event `ev` from happening (such as 2796 // following a link in the href of the a element, or doing a POST request 2797 // with page reload when a `<form>` is submitted). 2798 // Use it inside listener functions. 2799 function preventDefault(e) { 2800 if (e.preventDefault) { 2801 e.preventDefault(); 2802 } else { 2803 e.returnValue = false; 2804 } 2805 return this; 2806 } 2807 2808 // @function stop(ev: DOMEvent): this 2809 // Does `stopPropagation` and `preventDefault` at the same time. 2810 function stop(e) { 2811 preventDefault(e); 2812 stopPropagation(e); 2813 return this; 2814 } 2815 2816 // @function getMousePosition(ev: DOMEvent, container?: HTMLElement): Point 2817 // Gets normalized mouse position from a DOM event relative to the 2818 // `container` (border excluded) or to the whole page if not specified. 2819 function getMousePosition(e, container) { 2820 if (!container) { 2821 return new Point(e.clientX, e.clientY); 2822 } 2823 2824 var scale = getScale(container), 2825 offset = scale.boundingClientRect; // left and top values are in page scale (like the event clientX/Y) 2826 2827 return new Point( 2828 // offset.left/top values are in page scale (like clientX/Y), 2829 // whereas clientLeft/Top (border width) values are the original values (before CSS scale applies). 2830 (e.clientX - offset.left) / scale.x - container.clientLeft, 2831 (e.clientY - offset.top) / scale.y - container.clientTop 2832 ); 2833 } 2834 2835 // Chrome on Win scrolls double the pixels as in other platforms (see #4538), 2836 // and Firefox scrolls device pixels, not CSS pixels 2837 var wheelPxFactor = 2838 (win && chrome) ? 2 * window.devicePixelRatio : 2839 gecko ? window.devicePixelRatio : 1; 2840 2841 // @function getWheelDelta(ev: DOMEvent): Number 2842 // Gets normalized wheel delta from a mousewheel DOM event, in vertical 2843 // pixels scrolled (negative if scrolling down). 2844 // Events from pointing devices without precise scrolling are mapped to 2845 // a best guess of 60 pixels. 2846 function getWheelDelta(e) { 2847 return (edge) ? e.wheelDeltaY / 2 : // Don't trust window-geometry-based delta 2848 (e.deltaY && e.deltaMode === 0) ? -e.deltaY / wheelPxFactor : // Pixels 2849 (e.deltaY && e.deltaMode === 1) ? -e.deltaY * 20 : // Lines 2850 (e.deltaY && e.deltaMode === 2) ? -e.deltaY * 60 : // Pages 2851 (e.deltaX || e.deltaZ) ? 0 : // Skip horizontal/depth wheel events 2852 e.wheelDelta ? (e.wheelDeltaY || e.wheelDelta) / 2 : // Legacy IE pixels 2853 (e.detail && Math.abs(e.detail) < 32765) ? -e.detail * 20 : // Legacy Moz lines 2854 e.detail ? e.detail / -32765 * 60 : // Legacy Moz pages 2855 0; 2856 } 2857 2858 var skipEvents = {}; 2859 2860 function fakeStop(e) { 2861 // fakes stopPropagation by setting a special event flag, checked/reset with skipped(e) 2862 skipEvents[e.type] = true; 2863 } 2864 2865 function skipped(e) { 2866 var events = skipEvents[e.type]; 2867 // reset when checking, as it's only used in map container and propagates outside of the map 2868 skipEvents[e.type] = false; 2869 return events; 2870 } 2871 2872 // check if element really left/entered the event target (for mouseenter/mouseleave) 2873 function isExternalTarget(el, e) { 2874 2875 var related = e.relatedTarget; 2876 2877 if (!related) { return true; } 2878 2879 try { 2880 while (related && (related !== el)) { 2881 related = related.parentNode; 2882 } 2883 } catch (err) { 2884 return false; 2885 } 2886 return (related !== el); 2887 } 2888 2889 var lastClick; 2890 2891 // this is a horrible workaround for a bug in Android where a single touch triggers two click events 2892 function filterClick(e, handler) { 2893 var timeStamp = (e.timeStamp || (e.originalEvent && e.originalEvent.timeStamp)), 2894 elapsed = lastClick && (timeStamp - lastClick); 2895 2896 // are they closer together than 500ms yet more than 100ms? 2897 // Android typically triggers them ~300ms apart while multiple listeners 2898 // on the same event should be triggered far faster; 2899 // or check if click is simulated on the element, and if it is, reject any non-simulated events 2900 2901 if ((elapsed && elapsed > 100 && elapsed < 500) || (e.target._simulatedClick && !e._simulated)) { 2902 stop(e); 2903 return; 2904 } 2905 lastClick = timeStamp; 2906 2907 handler(e); 2908 } 2909 2910 2911 2912 2913 var DomEvent = (Object.freeze || Object)({ 2914 on: on, 2915 off: off, 2916 stopPropagation: stopPropagation, 2917 disableScrollPropagation: disableScrollPropagation, 2918 disableClickPropagation: disableClickPropagation, 2919 preventDefault: preventDefault, 2920 stop: stop, 2921 getMousePosition: getMousePosition, 2922 getWheelDelta: getWheelDelta, 2923 fakeStop: fakeStop, 2924 skipped: skipped, 2925 isExternalTarget: isExternalTarget, 2926 addListener: on, 2927 removeListener: off 2928 }); 2929 2930 /* 2931 * @class PosAnimation 2932 * @aka L.PosAnimation 2933 * @inherits Evented 2934 * Used internally for panning animations, utilizing CSS3 Transitions for modern browsers and a timer fallback for IE6-9. 2935 * 2936 * @example 2937 * ```js 2938 * var fx = new L.PosAnimation(); 2939 * fx.run(el, [300, 500], 0.5); 2940 * ``` 2941 * 2942 * @constructor L.PosAnimation() 2943 * Creates a `PosAnimation` object. 2944 * 2945 */ 2946 2947 var PosAnimation = Evented.extend({ 2948 2949 // @method run(el: HTMLElement, newPos: Point, duration?: Number, easeLinearity?: Number) 2950 // Run an animation of a given element to a new position, optionally setting 2951 // duration in seconds (`0.25` by default) and easing linearity factor (3rd 2952 // argument of the [cubic bezier curve](http://cubic-bezier.com/#0,0,.5,1), 2953 // `0.5` by default). 2954 run: function (el, newPos, duration, easeLinearity) { 2955 this.stop(); 2956 2957 this._el = el; 2958 this._inProgress = true; 2959 this._duration = duration || 0.25; 2960 this._easeOutPower = 1 / Math.max(easeLinearity || 0.5, 0.2); 2961 2962 this._startPos = getPosition(el); 2963 this._offset = newPos.subtract(this._startPos); 2964 this._startTime = +new Date(); 2965 2966 // @event start: Event 2967 // Fired when the animation starts 2968 this.fire('start'); 2969 2970 this._animate(); 2971 }, 2972 2973 // @method stop() 2974 // Stops the animation (if currently running). 2975 stop: function () { 2976 if (!this._inProgress) { return; } 2977 2978 this._step(true); 2979 this._complete(); 2980 }, 2981 2982 _animate: function () { 2983 // animation loop 2984 this._animId = requestAnimFrame(this._animate, this); 2985 this._step(); 2986 }, 2987 2988 _step: function (round) { 2989 var elapsed = (+new Date()) - this._startTime, 2990 duration = this._duration * 1000; 2991 2992 if (elapsed < duration) { 2993 this._runFrame(this._easeOut(elapsed / duration), round); 2994 } else { 2995 this._runFrame(1); 2996 this._complete(); 2997 } 2998 }, 2999 3000 _runFrame: function (progress, round) { 3001 var pos = this._startPos.add(this._offset.multiplyBy(progress)); 3002 if (round) { 3003 pos._round(); 3004 } 3005 setPosition(this._el, pos); 3006 3007 // @event step: Event 3008 // Fired continuously during the animation. 3009 this.fire('step'); 3010 }, 3011 3012 _complete: function () { 3013 cancelAnimFrame(this._animId); 3014 3015 this._inProgress = false; 3016 // @event end: Event 3017 // Fired when the animation ends. 3018 this.fire('end'); 3019 }, 3020 3021 _easeOut: function (t) { 3022 return 1 - Math.pow(1 - t, this._easeOutPower); 3023 } 3024 }); 3025 3026 /* 3027 * @class Map 3028 * @aka L.Map 3029 * @inherits Evented 3030 * 3031 * The central class of the API — it is used to create a map on a page and manipulate it. 3032 * 3033 * @example 3034 * 3035 * ```js 3036 * // initialize the map on the "map" div with a given center and zoom 3037 * var map = L.map('map', { 3038 * center: [51.505, -0.09], 3039 * zoom: 13 3040 * }); 3041 * ``` 3042 * 3043 */ 3044 3045 var Map = Evented.extend({ 3046 3047 options: { 3048 // @section Map State Options 3049 // @option crs: CRS = L.CRS.EPSG3857 3050 // The [Coordinate Reference System](#crs) to use. Don't change this if you're not 3051 // sure what it means. 3052 crs: EPSG3857, 3053 3054 // @option center: LatLng = undefined 3055 // Initial geographic center of the map 3056 center: undefined, 3057 3058 // @option zoom: Number = undefined 3059 // Initial map zoom level 3060 zoom: undefined, 3061 3062 // @option minZoom: Number = * 3063 // Minimum zoom level of the map. 3064 // If not specified and at least one `GridLayer` or `TileLayer` is in the map, 3065 // the lowest of their `minZoom` options will be used instead. 3066 minZoom: undefined, 3067 3068 // @option maxZoom: Number = * 3069 // Maximum zoom level of the map. 3070 // If not specified and at least one `GridLayer` or `TileLayer` is in the map, 3071 // the highest of their `maxZoom` options will be used instead. 3072 maxZoom: undefined, 3073 3074 // @option layers: Layer[] = [] 3075 // Array of layers that will be added to the map initially 3076 layers: [], 3077 3078 // @option maxBounds: LatLngBounds = null 3079 // When this option is set, the map restricts the view to the given 3080 // geographical bounds, bouncing the user back if the user tries to pan 3081 // outside the view. To set the restriction dynamically, use 3082 // [`setMaxBounds`](#map-setmaxbounds) method. 3083 maxBounds: undefined, 3084 3085 // @option renderer: Renderer = * 3086 // The default method for drawing vector layers on the map. `L.SVG` 3087 // or `L.Canvas` by default depending on browser support. 3088 renderer: undefined, 3089 3090 3091 // @section Animation Options 3092 // @option zoomAnimation: Boolean = true 3093 // Whether the map zoom animation is enabled. By default it's enabled 3094 // in all browsers that support CSS3 Transitions except Android. 3095 zoomAnimation: true, 3096 3097 // @option zoomAnimationThreshold: Number = 4 3098 // Won't animate zoom if the zoom difference exceeds this value. 3099 zoomAnimationThreshold: 4, 3100 3101 // @option fadeAnimation: Boolean = true 3102 // Whether the tile fade animation is enabled. By default it's enabled 3103 // in all browsers that support CSS3 Transitions except Android. 3104 fadeAnimation: true, 3105 3106 // @option markerZoomAnimation: Boolean = true 3107 // Whether markers animate their zoom with the zoom animation, if disabled 3108 // they will disappear for the length of the animation. By default it's 3109 // enabled in all browsers that support CSS3 Transitions except Android. 3110 markerZoomAnimation: true, 3111 3112 // @option transform3DLimit: Number = 2^23 3113 // Defines the maximum size of a CSS translation transform. The default 3114 // value should not be changed unless a web browser positions layers in 3115 // the wrong place after doing a large `panBy`. 3116 transform3DLimit: 8388608, // Precision limit of a 32-bit float 3117 3118 // @section Interaction Options 3119 // @option zoomSnap: Number = 1 3120 // Forces the map's zoom level to always be a multiple of this, particularly 3121 // right after a [`fitBounds()`](#map-fitbounds) or a pinch-zoom. 3122 // By default, the zoom level snaps to the nearest integer; lower values 3123 // (e.g. `0.5` or `0.1`) allow for greater granularity. A value of `0` 3124 // means the zoom level will not be snapped after `fitBounds` or a pinch-zoom. 3125 zoomSnap: 1, 3126 3127 // @option zoomDelta: Number = 1 3128 // Controls how much the map's zoom level will change after a 3129 // [`zoomIn()`](#map-zoomin), [`zoomOut()`](#map-zoomout), pressing `+` 3130 // or `-` on the keyboard, or using the [zoom controls](#control-zoom). 3131 // Values smaller than `1` (e.g. `0.5`) allow for greater granularity. 3132 zoomDelta: 1, 3133 3134 // @option trackResize: Boolean = true 3135 // Whether the map automatically handles browser window resize to update itself. 3136 trackResize: true 3137 }, 3138 3139 initialize: function (id, options) { // (HTMLElement or String, Object) 3140 options = setOptions(this, options); 3141 3142 // Make sure to assign internal flags at the beginning, 3143 // to avoid inconsistent state in some edge cases. 3144 this._handlers = []; 3145 this._layers = {}; 3146 this._zoomBoundLayers = {}; 3147 this._sizeChanged = true; 3148 3149 this._initContainer(id); 3150 this._initLayout(); 3151 3152 // hack for https://github.com/Leaflet/Leaflet/issues/1980 3153 this._onResize = bind(this._onResize, this); 3154 3155 this._initEvents(); 3156 3157 if (options.maxBounds) { 3158 this.setMaxBounds(options.maxBounds); 3159 } 3160 3161 if (options.zoom !== undefined) { 3162 this._zoom = this._limitZoom(options.zoom); 3163 } 3164 3165 if (options.center && options.zoom !== undefined) { 3166 this.setView(toLatLng(options.center), options.zoom, {reset: true}); 3167 } 3168 3169 this.callInitHooks(); 3170 3171 // don't animate on browsers without hardware-accelerated transitions or old Android/Opera 3172 this._zoomAnimated = TRANSITION && any3d && !mobileOpera && 3173 this.options.zoomAnimation; 3174 3175 // zoom transitions run with the same duration for all layers, so if one of transitionend events 3176 // happens after starting zoom animation (propagating to the map pane), we know that it ended globally 3177 if (this._zoomAnimated) { 3178 this._createAnimProxy(); 3179 on(this._proxy, TRANSITION_END, this._catchTransitionEnd, this); 3180 } 3181 3182 this._addLayers(this.options.layers); 3183 }, 3184 3185 3186 // @section Methods for modifying map state 3187 3188 // @method setView(center: LatLng, zoom: Number, options?: Zoom/pan options): this 3189 // Sets the view of the map (geographical center and zoom) with the given 3190 // animation options. 3191 setView: function (center, zoom, options) { 3192 3193 zoom = zoom === undefined ? this._zoom : this._limitZoom(zoom); 3194 center = this._limitCenter(toLatLng(center), zoom, this.options.maxBounds); 3195 options = options || {}; 3196 3197 this._stop(); 3198 3199 if (this._loaded && !options.reset && options !== true) { 3200 3201 if (options.animate !== undefined) { 3202 options.zoom = extend({animate: options.animate}, options.zoom); 3203 options.pan = extend({animate: options.animate, duration: options.duration}, options.pan); 3204 } 3205 3206 // try animating pan or zoom 3207 var moved = (this._zoom !== zoom) ? 3208 this._tryAnimatedZoom && this._tryAnimatedZoom(center, zoom, options.zoom) : 3209 this._tryAnimatedPan(center, options.pan); 3210 3211 if (moved) { 3212 // prevent resize handler call, the view will refresh after animation anyway 3213 clearTimeout(this._sizeTimer); 3214 return this; 3215 } 3216 } 3217 3218 // animation didn't start, just reset the map view 3219 this._resetView(center, zoom); 3220 3221 return this; 3222 }, 3223 3224 // @method setZoom(zoom: Number, options?: Zoom/pan options): this 3225 // Sets the zoom of the map. 3226 setZoom: function (zoom, options) { 3227 if (!this._loaded) { 3228 this._zoom = zoom; 3229 return this; 3230 } 3231 return this.setView(this.getCenter(), zoom, {zoom: options}); 3232 }, 3233 3234 // @method zoomIn(delta?: Number, options?: Zoom options): this 3235 // Increases the zoom of the map by `delta` ([`zoomDelta`](#map-zoomdelta) by default). 3236 zoomIn: function (delta, options) { 3237 delta = delta || (any3d ? this.options.zoomDelta : 1); 3238 return this.setZoom(this._zoom + delta, options); 3239 }, 3240 3241 // @method zoomOut(delta?: Number, options?: Zoom options): this 3242 // Decreases the zoom of the map by `delta` ([`zoomDelta`](#map-zoomdelta) by default). 3243 zoomOut: function (delta, options) { 3244 delta = delta || (any3d ? this.options.zoomDelta : 1); 3245 return this.setZoom(this._zoom - delta, options); 3246 }, 3247 3248 // @method setZoomAround(latlng: LatLng, zoom: Number, options: Zoom options): this 3249 // Zooms the map while keeping a specified geographical point on the map 3250 // stationary (e.g. used internally for scroll zoom and double-click zoom). 3251 // @alternative 3252 // @method setZoomAround(offset: Point, zoom: Number, options: Zoom options): this 3253 // Zooms the map while keeping a specified pixel on the map (relative to the top-left corner) stationary. 3254 setZoomAround: function (latlng, zoom, options) { 3255 var scale = this.getZoomScale(zoom), 3256 viewHalf = this.getSize().divideBy(2), 3257 containerPoint = latlng instanceof Point ? latlng : this.latLngToContainerPoint(latlng), 3258 3259 centerOffset = containerPoint.subtract(viewHalf).multiplyBy(1 - 1 / scale), 3260 newCenter = this.containerPointToLatLng(viewHalf.add(centerOffset)); 3261 3262 return this.setView(newCenter, zoom, {zoom: options}); 3263 }, 3264 3265 _getBoundsCenterZoom: function (bounds, options) { 3266 3267 options = options || {}; 3268 bounds = bounds.getBounds ? bounds.getBounds() : toLatLngBounds(bounds); 3269 3270 var paddingTL = toPoint(options.paddingTopLeft || options.padding || [0, 0]), 3271 paddingBR = toPoint(options.paddingBottomRight || options.padding || [0, 0]), 3272 3273 zoom = this.getBoundsZoom(bounds, false, paddingTL.add(paddingBR)); 3274 3275 zoom = (typeof options.maxZoom === 'number') ? Math.min(options.maxZoom, zoom) : zoom; 3276 3277 if (zoom === Infinity) { 3278 return { 3279 center: bounds.getCenter(), 3280 zoom: zoom 3281 }; 3282 } 3283 3284 var paddingOffset = paddingBR.subtract(paddingTL).divideBy(2), 3285 3286 swPoint = this.project(bounds.getSouthWest(), zoom), 3287 nePoint = this.project(bounds.getNorthEast(), zoom), 3288 center = this.unproject(swPoint.add(nePoint).divideBy(2).add(paddingOffset), zoom); 3289 3290 return { 3291 center: center, 3292 zoom: zoom 3293 }; 3294 }, 3295 3296 // @method fitBounds(bounds: LatLngBounds, options?: fitBounds options): this 3297 // Sets a map view that contains the given geographical bounds with the 3298 // maximum zoom level possible. 3299 fitBounds: function (bounds, options) { 3300 3301 bounds = toLatLngBounds(bounds); 3302 3303 if (!bounds.isValid()) { 3304 throw new Error('Bounds are not valid.'); 3305 } 3306 3307 var target = this._getBoundsCenterZoom(bounds, options); 3308 return this.setView(target.center, target.zoom, options); 3309 }, 3310 3311 // @method fitWorld(options?: fitBounds options): this 3312 // Sets a map view that mostly contains the whole world with the maximum 3313 // zoom level possible. 3314 fitWorld: function (options) { 3315 return this.fitBounds([[-90, -180], [90, 180]], options); 3316 }, 3317 3318 // @method panTo(latlng: LatLng, options?: Pan options): this 3319 // Pans the map to a given center. 3320 panTo: function (center, options) { // (LatLng) 3321 return this.setView(center, this._zoom, {pan: options}); 3322 }, 3323 3324 // @method panBy(offset: Point, options?: Pan options): this 3325 // Pans the map by a given number of pixels (animated). 3326 panBy: function (offset, options) { 3327 offset = toPoint(offset).round(); 3328 options = options || {}; 3329 3330 if (!offset.x && !offset.y) { 3331 return this.fire('moveend'); 3332 } 3333 // If we pan too far, Chrome gets issues with tiles 3334 // and makes them disappear or appear in the wrong place (slightly offset) #2602 3335 if (options.animate !== true && !this.getSize().contains(offset)) { 3336 this._resetView(this.unproject(this.project(this.getCenter()).add(offset)), this.getZoom()); 3337 return this; 3338 } 3339 3340 if (!this._panAnim) { 3341 this._panAnim = new PosAnimation(); 3342 3343 this._panAnim.on({ 3344 'step': this._onPanTransitionStep, 3345 'end': this._onPanTransitionEnd 3346 }, this); 3347 } 3348 3349 // don't fire movestart if animating inertia 3350 if (!options.noMoveStart) { 3351 this.fire('movestart'); 3352 } 3353 3354 // animate pan unless animate: false specified 3355 if (options.animate !== false) { 3356 addClass(this._mapPane, 'leaflet-pan-anim'); 3357 3358 var newPos = this._getMapPanePos().subtract(offset).round(); 3359 this._panAnim.run(this._mapPane, newPos, options.duration || 0.25, options.easeLinearity); 3360 } else { 3361 this._rawPanBy(offset); 3362 this.fire('move').fire('moveend'); 3363 } 3364 3365 return this; 3366 }, 3367 3368 // @method flyTo(latlng: LatLng, zoom?: Number, options?: Zoom/pan options): this 3369 // Sets the view of the map (geographical center and zoom) performing a smooth 3370 // pan-zoom animation. 3371 flyTo: function (targetCenter, targetZoom, options) { 3372 3373 options = options || {}; 3374 if (options.animate === false || !any3d) { 3375 return this.setView(targetCenter, targetZoom, options); 3376 } 3377 3378 this._stop(); 3379 3380 var from = this.project(this.getCenter()), 3381 to = this.project(targetCenter), 3382 size = this.getSize(), 3383 startZoom = this._zoom; 3384 3385 targetCenter = toLatLng(targetCenter); 3386 targetZoom = targetZoom === undefined ? startZoom : targetZoom; 3387 3388 var w0 = Math.max(size.x, size.y), 3389 w1 = w0 * this.getZoomScale(startZoom, targetZoom), 3390 u1 = (to.distanceTo(from)) || 1, 3391 rho = 1.42, 3392 rho2 = rho * rho; 3393 3394 function r(i) { 3395 var s1 = i ? -1 : 1, 3396 s2 = i ? w1 : w0, 3397 t1 = w1 * w1 - w0 * w0 + s1 * rho2 * rho2 * u1 * u1, 3398 b1 = 2 * s2 * rho2 * u1, 3399 b = t1 / b1, 3400 sq = Math.sqrt(b * b + 1) - b; 3401 3402 // workaround for floating point precision bug when sq = 0, log = -Infinite, 3403 // thus triggering an infinite loop in flyTo 3404 var log = sq < 0.000000001 ? -18 : Math.log(sq); 3405 3406 return log; 3407 } 3408 3409 function sinh(n) { return (Math.exp(n) - Math.exp(-n)) / 2; } 3410 function cosh(n) { return (Math.exp(n) + Math.exp(-n)) / 2; } 3411 function tanh(n) { return sinh(n) / cosh(n); } 3412 3413 var r0 = r(0); 3414 3415 function w(s) { return w0 * (cosh(r0) / cosh(r0 + rho * s)); } 3416 function u(s) { return w0 * (cosh(r0) * tanh(r0 + rho * s) - sinh(r0)) / rho2; } 3417 3418 function easeOut(t) { return 1 - Math.pow(1 - t, 1.5); } 3419 3420 var start = Date.now(), 3421 S = (r(1) - r0) / rho, 3422 duration = options.duration ? 1000 * options.duration : 1000 * S * 0.8; 3423 3424 function frame() { 3425 var t = (Date.now() - start) / duration, 3426 s = easeOut(t) * S; 3427 3428 if (t <= 1) { 3429 this._flyToFrame = requestAnimFrame(frame, this); 3430 3431 this._move( 3432 this.unproject(from.add(to.subtract(from).multiplyBy(u(s) / u1)), startZoom), 3433 this.getScaleZoom(w0 / w(s), startZoom), 3434 {flyTo: true}); 3435 3436 } else { 3437 this 3438 ._move(targetCenter, targetZoom) 3439 ._moveEnd(true); 3440 } 3441 } 3442 3443 this._moveStart(true, options.noMoveStart); 3444 3445 frame.call(this); 3446 return this; 3447 }, 3448 3449 // @method flyToBounds(bounds: LatLngBounds, options?: fitBounds options): this 3450 // Sets the view of the map with a smooth animation like [`flyTo`](#map-flyto), 3451 // but takes a bounds parameter like [`fitBounds`](#map-fitbounds). 3452 flyToBounds: function (bounds, options) { 3453 var target = this._getBoundsCenterZoom(bounds, options); 3454 return this.flyTo(target.center, target.zoom, options); 3455 }, 3456 3457 // @method setMaxBounds(bounds: Bounds): this 3458 // Restricts the map view to the given bounds (see the [maxBounds](#map-maxbounds) option). 3459 setMaxBounds: function (bounds) { 3460 bounds = toLatLngBounds(bounds); 3461 3462 if (!bounds.isValid()) { 3463 this.options.maxBounds = null; 3464 return this.off('moveend', this._panInsideMaxBounds); 3465 } else if (this.options.maxBounds) { 3466 this.off('moveend', this._panInsideMaxBounds); 3467 } 3468 3469 this.options.maxBounds = bounds; 3470 3471 if (this._loaded) { 3472 this._panInsideMaxBounds(); 3473 } 3474 3475 return this.on('moveend', this._panInsideMaxBounds); 3476 }, 3477 3478 // @method setMinZoom(zoom: Number): this 3479 // Sets the lower limit for the available zoom levels (see the [minZoom](#map-minzoom) option). 3480 setMinZoom: function (zoom) { 3481 var oldZoom = this.options.minZoom; 3482 this.options.minZoom = zoom; 3483 3484 if (this._loaded && oldZoom !== zoom) { 3485 this.fire('zoomlevelschange'); 3486 3487 if (this.getZoom() < this.options.minZoom) { 3488 return this.setZoom(zoom); 3489 } 3490 } 3491 3492 return this; 3493 }, 3494 3495 // @method setMaxZoom(zoom: Number): this 3496 // Sets the upper limit for the available zoom levels (see the [maxZoom](#map-maxzoom) option). 3497 setMaxZoom: function (zoom) { 3498 var oldZoom = this.options.maxZoom; 3499 this.options.maxZoom = zoom; 3500 3501 if (this._loaded && oldZoom !== zoom) { 3502 this.fire('zoomlevelschange'); 3503 3504 if (this.getZoom() > this.options.maxZoom) { 3505 return this.setZoom(zoom); 3506 } 3507 } 3508 3509 return this; 3510 }, 3511 3512 // @method panInsideBounds(bounds: LatLngBounds, options?: Pan options): this 3513 // Pans the map to the closest view that would lie inside the given bounds (if it's not already), controlling the animation using the options specific, if any. 3514 panInsideBounds: function (bounds, options) { 3515 this._enforcingBounds = true; 3516 var center = this.getCenter(), 3517 newCenter = this._limitCenter(center, this._zoom, toLatLngBounds(bounds)); 3518 3519 if (!center.equals(newCenter)) { 3520 this.panTo(newCenter, options); 3521 } 3522 3523 this._enforcingBounds = false; 3524 return this; 3525 }, 3526 3527 // @method panInside(latlng: LatLng, options?: options): this 3528 // Pans the map the minimum amount to make the `latlng` visible. Use 3529 // `padding`, `paddingTopLeft` and `paddingTopRight` options to fit 3530 // the display to more restricted bounds, like [`fitBounds`](#map-fitbounds). 3531 // If `latlng` is already within the (optionally padded) display bounds, 3532 // the map will not be panned. 3533 panInside: function (latlng, options) { 3534 options = options || {}; 3535 3536 var paddingTL = toPoint(options.paddingTopLeft || options.padding || [0, 0]), 3537 paddingBR = toPoint(options.paddingBottomRight || options.padding || [0, 0]), 3538 center = this.getCenter(), 3539 pixelCenter = this.project(center), 3540 pixelPoint = this.project(latlng), 3541 pixelBounds = this.getPixelBounds(), 3542 halfPixelBounds = pixelBounds.getSize().divideBy(2), 3543 paddedBounds = toBounds([pixelBounds.min.add(paddingTL), pixelBounds.max.subtract(paddingBR)]); 3544 3545 if (!paddedBounds.contains(pixelPoint)) { 3546 this._enforcingBounds = true; 3547 var diff = pixelCenter.subtract(pixelPoint), 3548 newCenter = toPoint(pixelPoint.x + diff.x, pixelPoint.y + diff.y); 3549 3550 if (pixelPoint.x < paddedBounds.min.x || pixelPoint.x > paddedBounds.max.x) { 3551 newCenter.x = pixelCenter.x - diff.x; 3552 if (diff.x > 0) { 3553 newCenter.x += halfPixelBounds.x - paddingTL.x; 3554 } else { 3555 newCenter.x -= halfPixelBounds.x - paddingBR.x; 3556 } 3557 } 3558 if (pixelPoint.y < paddedBounds.min.y || pixelPoint.y > paddedBounds.max.y) { 3559 newCenter.y = pixelCenter.y - diff.y; 3560 if (diff.y > 0) { 3561 newCenter.y += halfPixelBounds.y - paddingTL.y; 3562 } else { 3563 newCenter.y -= halfPixelBounds.y - paddingBR.y; 3564 } 3565 } 3566 this.panTo(this.unproject(newCenter), options); 3567 this._enforcingBounds = false; 3568 } 3569 return this; 3570 }, 3571 3572 // @method invalidateSize(options: Zoom/pan options): this 3573 // Checks if the map container size changed and updates the map if so — 3574 // call it after you've changed the map size dynamically, also animating 3575 // pan by default. If `options.pan` is `false`, panning will not occur. 3576 // If `options.debounceMoveend` is `true`, it will delay `moveend` event so 3577 // that it doesn't happen often even if the method is called many 3578 // times in a row. 3579 3580 // @alternative 3581 // @method invalidateSize(animate: Boolean): this 3582 // Checks if the map container size changed and updates the map if so — 3583 // call it after you've changed the map size dynamically, also animating 3584 // pan by default. 3585 invalidateSize: function (options) { 3586 if (!this._loaded) { return this; } 3587 3588 options = extend({ 3589 animate: false, 3590 pan: true 3591 }, options === true ? {animate: true} : options); 3592 3593 var oldSize = this.getSize(); 3594 this._sizeChanged = true; 3595 this._lastCenter = null; 3596 3597 var newSize = this.getSize(), 3598 oldCenter = oldSize.divideBy(2).round(), 3599 newCenter = newSize.divideBy(2).round(), 3600 offset = oldCenter.subtract(newCenter); 3601 3602 if (!offset.x && !offset.y) { return this; } 3603 3604 if (options.animate && options.pan) { 3605 this.panBy(offset); 3606 3607 } else { 3608 if (options.pan) { 3609 this._rawPanBy(offset); 3610 } 3611 3612 this.fire('move'); 3613 3614 if (options.debounceMoveend) { 3615 clearTimeout(this._sizeTimer); 3616 this._sizeTimer = setTimeout(bind(this.fire, this, 'moveend'), 200); 3617 } else { 3618 this.fire('moveend'); 3619 } 3620 } 3621 3622 // @section Map state change events 3623 // @event resize: ResizeEvent 3624 // Fired when the map is resized. 3625 return this.fire('resize', { 3626 oldSize: oldSize, 3627 newSize: newSize 3628 }); 3629 }, 3630 3631 // @section Methods for modifying map state 3632 // @method stop(): this 3633 // Stops the currently running `panTo` or `flyTo` animation, if any. 3634 stop: function () { 3635 this.setZoom(this._limitZoom(this._zoom)); 3636 if (!this.options.zoomSnap) { 3637 this.fire('viewreset'); 3638 } 3639 return this._stop(); 3640 }, 3641 3642 // @section Geolocation methods 3643 // @method locate(options?: Locate options): this 3644 // Tries to locate the user using the Geolocation API, firing a [`locationfound`](#map-locationfound) 3645 // event with location data on success or a [`locationerror`](#map-locationerror) event on failure, 3646 // and optionally sets the map view to the user's location with respect to 3647 // detection accuracy (or to the world view if geolocation failed). 3648 // Note that, if your page doesn't use HTTPS, this method will fail in 3649 // modern browsers ([Chrome 50 and newer](https://sites.google.com/a/chromium.org/dev/Home/chromium-security/deprecating-powerful-features-on-insecure-origins)) 3650 // See `Locate options` for more details. 3651 locate: function (options) { 3652 3653 options = this._locateOptions = extend({ 3654 timeout: 10000, 3655 watch: false 3656 // setView: false 3657 // maxZoom: <Number> 3658 // maximumAge: 0 3659 // enableHighAccuracy: false 3660 }, options); 3661 3662 if (!('geolocation' in navigator)) { 3663 this._handleGeolocationError({ 3664 code: 0, 3665 message: 'Geolocation not supported.' 3666 }); 3667 return this; 3668 } 3669 3670 var onResponse = bind(this._handleGeolocationResponse, this), 3671 onError = bind(this._handleGeolocationError, this); 3672 3673 if (options.watch) { 3674 this._locationWatchId = 3675 navigator.geolocation.watchPosition(onResponse, onError, options); 3676 } else { 3677 navigator.geolocation.getCurrentPosition(onResponse, onError, options); 3678 } 3679 return this; 3680 }, 3681 3682 // @method stopLocate(): this 3683 // Stops watching location previously initiated by `map.locate({watch: true})` 3684 // and aborts resetting the map view if map.locate was called with 3685 // `{setView: true}`. 3686 stopLocate: function () { 3687 if (navigator.geolocation && navigator.geolocation.clearWatch) { 3688 navigator.geolocation.clearWatch(this._locationWatchId); 3689 } 3690 if (this._locateOptions) { 3691 this._locateOptions.setView = false; 3692 } 3693 return this; 3694 }, 3695 3696 _handleGeolocationError: function (error) { 3697 var c = error.code, 3698 message = error.message || 3699 (c === 1 ? 'permission denied' : 3700 (c === 2 ? 'position unavailable' : 'timeout')); 3701 3702 if (this._locateOptions.setView && !this._loaded) { 3703 this.fitWorld(); 3704 } 3705 3706 // @section Location events 3707 // @event locationerror: ErrorEvent 3708 // Fired when geolocation (using the [`locate`](#map-locate) method) failed. 3709 this.fire('locationerror', { 3710 code: c, 3711 message: 'Geolocation error: ' + message + '.' 3712 }); 3713 }, 3714 3715 _handleGeolocationResponse: function (pos) { 3716 var lat = pos.coords.latitude, 3717 lng = pos.coords.longitude, 3718 latlng = new LatLng(lat, lng), 3719 bounds = latlng.toBounds(pos.coords.accuracy * 2), 3720 options = this._locateOptions; 3721 3722 if (options.setView) { 3723 var zoom = this.getBoundsZoom(bounds); 3724 this.setView(latlng, options.maxZoom ? Math.min(zoom, options.maxZoom) : zoom); 3725 } 3726 3727 var data = { 3728 latlng: latlng, 3729 bounds: bounds, 3730 timestamp: pos.timestamp 3731 }; 3732 3733 for (var i in pos.coords) { 3734 if (typeof pos.coords[i] === 'number') { 3735 data[i] = pos.coords[i]; 3736 } 3737 } 3738 3739 // @event locationfound: LocationEvent 3740 // Fired when geolocation (using the [`locate`](#map-locate) method) 3741 // went successfully. 3742 this.fire('locationfound', data); 3743 }, 3744 3745 // TODO Appropriate docs section? 3746 // @section Other Methods 3747 // @method addHandler(name: String, HandlerClass: Function): this 3748 // Adds a new `Handler` to the map, given its name and constructor function. 3749 addHandler: function (name, HandlerClass) { 3750 if (!HandlerClass) { return this; } 3751 3752 var handler = this[name] = new HandlerClass(this); 3753 3754 this._handlers.push(handler); 3755 3756 if (this.options[name]) { 3757 handler.enable(); 3758 } 3759 3760 return this; 3761 }, 3762 3763 // @method remove(): this 3764 // Destroys the map and clears all related event listeners. 3765 remove: function () { 3766 3767 this._initEvents(true); 3768 3769 if (this._containerId !== this._container._leaflet_id) { 3770 throw new Error('Map container is being reused by another instance'); 3771 } 3772 3773 try { 3774 // throws error in IE6-8 3775 delete this._container._leaflet_id; 3776 delete this._containerId; 3777 } catch (e) { 3778 /*eslint-disable */ 3779 this._container._leaflet_id = undefined; 3780 /* eslint-enable */ 3781 this._containerId = undefined; 3782 } 3783 3784 if (this._locationWatchId !== undefined) { 3785 this.stopLocate(); 3786 } 3787 3788 this._stop(); 3789 3790 remove(this._mapPane); 3791 3792 if (this._clearControlPos) { 3793 this._clearControlPos(); 3794 } 3795 if (this._resizeRequest) { 3796 cancelAnimFrame(this._resizeRequest); 3797 this._resizeRequest = null; 3798 } 3799 3800 this._clearHandlers(); 3801 3802 if (this._loaded) { 3803 // @section Map state change events 3804 // @event unload: Event 3805 // Fired when the map is destroyed with [remove](#map-remove) method. 3806 this.fire('unload'); 3807 } 3808 3809 var i; 3810 for (i in this._layers) { 3811 this._layers[i].remove(); 3812 } 3813 for (i in this._panes) { 3814 remove(this._panes[i]); 3815 } 3816 3817 this._layers = []; 3818 this._panes = []; 3819 delete this._mapPane; 3820 delete this._renderer; 3821 3822 return this; 3823 }, 3824 3825 // @section Other Methods 3826 // @method createPane(name: String, container?: HTMLElement): HTMLElement 3827 // Creates a new [map pane](#map-pane) with the given name if it doesn't exist already, 3828 // then returns it. The pane is created as a child of `container`, or 3829 // as a child of the main map pane if not set. 3830 createPane: function (name, container) { 3831 var className = 'leaflet-pane' + (name ? ' leaflet-' + name.replace('Pane', '') + '-pane' : ''), 3832 pane = create$1('div', className, container || this._mapPane); 3833 3834 if (name) { 3835 this._panes[name] = pane; 3836 } 3837 return pane; 3838 }, 3839 3840 // @section Methods for Getting Map State 3841 3842 // @method getCenter(): LatLng 3843 // Returns the geographical center of the map view 3844 getCenter: function () { 3845 this._checkIfLoaded(); 3846 3847 if (this._lastCenter && !this._moved()) { 3848 return this._lastCenter; 3849 } 3850 return this.layerPointToLatLng(this._getCenterLayerPoint()); 3851 }, 3852 3853 // @method getZoom(): Number 3854 // Returns the current zoom level of the map view 3855 getZoom: function () { 3856 return this._zoom; 3857 }, 3858 3859 // @method getBounds(): LatLngBounds 3860 // Returns the geographical bounds visible in the current map view 3861 getBounds: function () { 3862 var bounds = this.getPixelBounds(), 3863 sw = this.unproject(bounds.getBottomLeft()), 3864 ne = this.unproject(bounds.getTopRight()); 3865 3866 return new LatLngBounds(sw, ne); 3867 }, 3868 3869 // @method getMinZoom(): Number 3870 // Returns the minimum zoom level of the map (if set in the `minZoom` option of the map or of any layers), or `0` by default. 3871 getMinZoom: function () { 3872 return this.options.minZoom === undefined ? this._layersMinZoom || 0 : this.options.minZoom; 3873 }, 3874 3875 // @method getMaxZoom(): Number 3876 // Returns the maximum zoom level of the map (if set in the `maxZoom` option of the map or of any layers). 3877 getMaxZoom: function () { 3878 return this.options.maxZoom === undefined ? 3879 (this._layersMaxZoom === undefined ? Infinity : this._layersMaxZoom) : 3880 this.options.maxZoom; 3881 }, 3882 3883 // @method getBoundsZoom(bounds: LatLngBounds, inside?: Boolean, padding?: Point): Number 3884 // Returns the maximum zoom level on which the given bounds fit to the map 3885 // view in its entirety. If `inside` (optional) is set to `true`, the method 3886 // instead returns the minimum zoom level on which the map view fits into 3887 // the given bounds in its entirety. 3888 getBoundsZoom: function (bounds, inside, padding) { // (LatLngBounds[, Boolean, Point]) -> Number 3889 bounds = toLatLngBounds(bounds); 3890 padding = toPoint(padding || [0, 0]); 3891 3892 var zoom = this.getZoom() || 0, 3893 min = this.getMinZoom(), 3894 max = this.getMaxZoom(), 3895 nw = bounds.getNorthWest(), 3896 se = bounds.getSouthEast(), 3897 size = this.getSize().subtract(padding), 3898 boundsSize = toBounds(this.project(se, zoom), this.project(nw, zoom)).getSize(), 3899 snap = any3d ? this.options.zoomSnap : 1, 3900 scalex = size.x / boundsSize.x, 3901 scaley = size.y / boundsSize.y, 3902 scale = inside ? Math.max(scalex, scaley) : Math.min(scalex, scaley); 3903 3904 zoom = this.getScaleZoom(scale, zoom); 3905 3906 if (snap) { 3907 zoom = Math.round(zoom / (snap / 100)) * (snap / 100); // don't jump if within 1% of a snap level 3908 zoom = inside ? Math.ceil(zoom / snap) * snap : Math.floor(zoom / snap) * snap; 3909 } 3910 3911 return Math.max(min, Math.min(max, zoom)); 3912 }, 3913 3914 // @method getSize(): Point 3915 // Returns the current size of the map container (in pixels). 3916 getSize: function () { 3917 if (!this._size || this._sizeChanged) { 3918 this._size = new Point( 3919 this._container.clientWidth || 0, 3920 this._container.clientHeight || 0); 3921 3922 this._sizeChanged = false; 3923 } 3924 return this._size.clone(); 3925 }, 3926 3927 // @method getPixelBounds(): Bounds 3928 // Returns the bounds of the current map view in projected pixel 3929 // coordinates (sometimes useful in layer and overlay implementations). 3930 getPixelBounds: function (center, zoom) { 3931 var topLeftPoint = this._getTopLeftPoint(center, zoom); 3932 return new Bounds(topLeftPoint, topLeftPoint.add(this.getSize())); 3933 }, 3934 3935 // TODO: Check semantics - isn't the pixel origin the 0,0 coord relative to 3936 // the map pane? "left point of the map layer" can be confusing, specially 3937 // since there can be negative offsets. 3938 // @method getPixelOrigin(): Point 3939 // Returns the projected pixel coordinates of the top left point of 3940 // the map layer (useful in custom layer and overlay implementations). 3941 getPixelOrigin: function () { 3942 this._checkIfLoaded(); 3943 return this._pixelOrigin; 3944 }, 3945 3946 // @method getPixelWorldBounds(zoom?: Number): Bounds 3947 // Returns the world's bounds in pixel coordinates for zoom level `zoom`. 3948 // If `zoom` is omitted, the map's current zoom level is used. 3949 getPixelWorldBounds: function (zoom) { 3950 return this.options.crs.getProjectedBounds(zoom === undefined ? this.getZoom() : zoom); 3951 }, 3952 3953 // @section Other Methods 3954 3955 // @method getPane(pane: String|HTMLElement): HTMLElement 3956 // Returns a [map pane](#map-pane), given its name or its HTML element (its identity). 3957 getPane: function (pane) { 3958 return typeof pane === 'string' ? this._panes[pane] : pane; 3959 }, 3960 3961 // @method getPanes(): Object 3962 // Returns a plain object containing the names of all [panes](#map-pane) as keys and 3963 // the panes as values. 3964 getPanes: function () { 3965 return this._panes; 3966 }, 3967 3968 // @method getContainer: HTMLElement 3969 // Returns the HTML element that contains the map. 3970 getContainer: function () { 3971 return this._container; 3972 }, 3973 3974 3975 // @section Conversion Methods 3976 3977 // @method getZoomScale(toZoom: Number, fromZoom: Number): Number 3978 // Returns the scale factor to be applied to a map transition from zoom level 3979 // `fromZoom` to `toZoom`. Used internally to help with zoom animations. 3980 getZoomScale: function (toZoom, fromZoom) { 3981 // TODO replace with universal implementation after refactoring projections 3982 var crs = this.options.crs; 3983 fromZoom = fromZoom === undefined ? this._zoom : fromZoom; 3984 return crs.scale(toZoom) / crs.scale(fromZoom); 3985 }, 3986 3987 // @method getScaleZoom(scale: Number, fromZoom: Number): Number 3988 // Returns the zoom level that the map would end up at, if it is at `fromZoom` 3989 // level and everything is scaled by a factor of `scale`. Inverse of 3990 // [`getZoomScale`](#map-getZoomScale). 3991 getScaleZoom: function (scale, fromZoom) { 3992 var crs = this.options.crs; 3993 fromZoom = fromZoom === undefined ? this._zoom : fromZoom; 3994 var zoom = crs.zoom(scale * crs.scale(fromZoom)); 3995 return isNaN(zoom) ? Infinity : zoom; 3996 }, 3997 3998 // @method project(latlng: LatLng, zoom: Number): Point 3999 // Projects a geographical coordinate `LatLng` according to the projection 4000 // of the map's CRS, then scales it according to `zoom` and the CRS's 4001 // `Transformation`. The result is pixel coordinate relative to 4002 // the CRS origin. 4003 project: function (latlng, zoom) { 4004 zoom = zoom === undefined ? this._zoom : zoom; 4005 return this.options.crs.latLngToPoint(toLatLng(latlng), zoom); 4006 }, 4007 4008 // @method unproject(point: Point, zoom: Number): LatLng 4009 // Inverse of [`project`](#map-project). 4010 unproject: function (point, zoom) { 4011 zoom = zoom === undefined ? this._zoom : zoom; 4012 return this.options.crs.pointToLatLng(toPoint(point), zoom); 4013 }, 4014 4015 // @method layerPointToLatLng(point: Point): LatLng 4016 // Given a pixel coordinate relative to the [origin pixel](#map-getpixelorigin), 4017 // returns the corresponding geographical coordinate (for the current zoom level). 4018 layerPointToLatLng: function (point) { 4019 var projectedPoint = toPoint(point).add(this.getPixelOrigin()); 4020 return this.unproject(projectedPoint); 4021 }, 4022 4023 // @method latLngToLayerPoint(latlng: LatLng): Point 4024 // Given a geographical coordinate, returns the corresponding pixel coordinate 4025 // relative to the [origin pixel](#map-getpixelorigin). 4026 latLngToLayerPoint: function (latlng) { 4027 var projectedPoint = this.project(toLatLng(latlng))._round(); 4028 return projectedPoint._subtract(this.getPixelOrigin()); 4029 }, 4030 4031 // @method wrapLatLng(latlng: LatLng): LatLng 4032 // Returns a `LatLng` where `lat` and `lng` has been wrapped according to the 4033 // map's CRS's `wrapLat` and `wrapLng` properties, if they are outside the 4034 // CRS's bounds. 4035 // By default this means longitude is wrapped around the dateline so its 4036 // value is between -180 and +180 degrees. 4037 wrapLatLng: function (latlng) { 4038 return this.options.crs.wrapLatLng(toLatLng(latlng)); 4039 }, 4040 4041 // @method wrapLatLngBounds(bounds: LatLngBounds): LatLngBounds 4042 // Returns a `LatLngBounds` with the same size as the given one, ensuring that 4043 // its center is within the CRS's bounds. 4044 // By default this means the center longitude is wrapped around the dateline so its 4045 // value is between -180 and +180 degrees, and the majority of the bounds 4046 // overlaps the CRS's bounds. 4047 wrapLatLngBounds: function (latlng) { 4048 return this.options.crs.wrapLatLngBounds(toLatLngBounds(latlng)); 4049 }, 4050 4051 // @method distance(latlng1: LatLng, latlng2: LatLng): Number 4052 // Returns the distance between two geographical coordinates according to 4053 // the map's CRS. By default this measures distance in meters. 4054 distance: function (latlng1, latlng2) { 4055 return this.options.crs.distance(toLatLng(latlng1), toLatLng(latlng2)); 4056 }, 4057 4058 // @method containerPointToLayerPoint(point: Point): Point 4059 // Given a pixel coordinate relative to the map container, returns the corresponding 4060 // pixel coordinate relative to the [origin pixel](#map-getpixelorigin). 4061 containerPointToLayerPoint: function (point) { // (Point) 4062 return toPoint(point).subtract(this._getMapPanePos()); 4063 }, 4064 4065 // @method layerPointToContainerPoint(point: Point): Point 4066 // Given a pixel coordinate relative to the [origin pixel](#map-getpixelorigin), 4067 // returns the corresponding pixel coordinate relative to the map container. 4068 layerPointToContainerPoint: function (point) { // (Point) 4069 return toPoint(point).add(this._getMapPanePos()); 4070 }, 4071 4072 // @method containerPointToLatLng(point: Point): LatLng 4073 // Given a pixel coordinate relative to the map container, returns 4074 // the corresponding geographical coordinate (for the current zoom level). 4075 containerPointToLatLng: function (point) { 4076 var layerPoint = this.containerPointToLayerPoint(toPoint(point)); 4077 return this.layerPointToLatLng(layerPoint); 4078 }, 4079 4080 // @method latLngToContainerPoint(latlng: LatLng): Point 4081 // Given a geographical coordinate, returns the corresponding pixel coordinate 4082 // relative to the map container. 4083 latLngToContainerPoint: function (latlng) { 4084 return this.layerPointToContainerPoint(this.latLngToLayerPoint(toLatLng(latlng))); 4085 }, 4086 4087 // @method mouseEventToContainerPoint(ev: MouseEvent): Point 4088 // Given a MouseEvent object, returns the pixel coordinate relative to the 4089 // map container where the event took place. 4090 mouseEventToContainerPoint: function (e) { 4091 return getMousePosition(e, this._container); 4092 }, 4093 4094 // @method mouseEventToLayerPoint(ev: MouseEvent): Point 4095 // Given a MouseEvent object, returns the pixel coordinate relative to 4096 // the [origin pixel](#map-getpixelorigin) where the event took place. 4097 mouseEventToLayerPoint: function (e) { 4098 return this.containerPointToLayerPoint(this.mouseEventToContainerPoint(e)); 4099 }, 4100 4101 // @method mouseEventToLatLng(ev: MouseEvent): LatLng 4102 // Given a MouseEvent object, returns geographical coordinate where the 4103 // event took place. 4104 mouseEventToLatLng: function (e) { // (MouseEvent) 4105 return this.layerPointToLatLng(this.mouseEventToLayerPoint(e)); 4106 }, 4107 4108 4109 // map initialization methods 4110 4111 _initContainer: function (id) { 4112 var container = this._container = get(id); 4113 4114 if (!container) { 4115 throw new Error('Map container not found.'); 4116 } else if (container._leaflet_id) { 4117 throw new Error('Map container is already initialized.'); 4118 } 4119 4120 on(container, 'scroll', this._onScroll, this); 4121 this._containerId = stamp(container); 4122 }, 4123 4124 _initLayout: function () { 4125 var container = this._container; 4126 4127 this._fadeAnimated = this.options.fadeAnimation && any3d; 4128 4129 addClass(container, 'leaflet-container' + 4130 (touch ? ' leaflet-touch' : '') + 4131 (retina ? ' leaflet-retina' : '') + 4132 (ielt9 ? ' leaflet-oldie' : '') + 4133 (safari ? ' leaflet-safari' : '') + 4134 (this._fadeAnimated ? ' leaflet-fade-anim' : '')); 4135 4136 var position = getStyle(container, 'position'); 4137 4138 if (position !== 'absolute' && position !== 'relative' && position !== 'fixed') { 4139 container.style.position = 'relative'; 4140 } 4141 4142 this._initPanes(); 4143 4144 if (this._initControlPos) { 4145 this._initControlPos(); 4146 } 4147 }, 4148 4149 _initPanes: function () { 4150 var panes = this._panes = {}; 4151 this._paneRenderers = {}; 4152 4153 // @section 4154 // 4155 // Panes are DOM elements used to control the ordering of layers on the map. You 4156 // can access panes with [`map.getPane`](#map-getpane) or 4157 // [`map.getPanes`](#map-getpanes) methods. New panes can be created with the 4158 // [`map.createPane`](#map-createpane) method. 4159 // 4160 // Every map has the following default panes that differ only in zIndex. 4161 // 4162 // @pane mapPane: HTMLElement = 'auto' 4163 // Pane that contains all other map panes 4164 4165 this._mapPane = this.createPane('mapPane', this._container); 4166 setPosition(this._mapPane, new Point(0, 0)); 4167 4168 // @pane tilePane: HTMLElement = 200 4169 // Pane for `GridLayer`s and `TileLayer`s 4170 this.createPane('tilePane'); 4171 // @pane overlayPane: HTMLElement = 400 4172 // Pane for vectors (`Path`s, like `Polyline`s and `Polygon`s), `ImageOverlay`s and `VideoOverlay`s 4173 this.createPane('shadowPane'); 4174 // @pane shadowPane: HTMLElement = 500 4175 // Pane for overlay shadows (e.g. `Marker` shadows) 4176 this.createPane('overlayPane'); 4177 // @pane markerPane: HTMLElement = 600 4178 // Pane for `Icon`s of `Marker`s 4179 this.createPane('markerPane'); 4180 // @pane tooltipPane: HTMLElement = 650 4181 // Pane for `Tooltip`s. 4182 this.createPane('tooltipPane'); 4183 // @pane popupPane: HTMLElement = 700 4184 // Pane for `Popup`s. 4185 this.createPane('popupPane'); 4186 4187 if (!this.options.markerZoomAnimation) { 4188 addClass(panes.markerPane, 'leaflet-zoom-hide'); 4189 addClass(panes.shadowPane, 'leaflet-zoom-hide'); 4190 } 4191 }, 4192 4193 4194 // private methods that modify map state 4195 4196 // @section Map state change events 4197 _resetView: function (center, zoom) { 4198 setPosition(this._mapPane, new Point(0, 0)); 4199 4200 var loading = !this._loaded; 4201 this._loaded = true; 4202 zoom = this._limitZoom(zoom); 4203 4204 this.fire('viewprereset'); 4205 4206 var zoomChanged = this._zoom !== zoom; 4207 this 4208 ._moveStart(zoomChanged, false) 4209 ._move(center, zoom) 4210 ._moveEnd(zoomChanged); 4211 4212 // @event viewreset: Event 4213 // Fired when the map needs to redraw its content (this usually happens 4214 // on map zoom or load). Very useful for creating custom overlays. 4215 this.fire('viewreset'); 4216 4217 // @event load: Event 4218 // Fired when the map is initialized (when its center and zoom are set 4219 // for the first time). 4220 if (loading) { 4221 this.fire('load'); 4222 } 4223 }, 4224 4225 _moveStart: function (zoomChanged, noMoveStart) { 4226 // @event zoomstart: Event 4227 // Fired when the map zoom is about to change (e.g. before zoom animation). 4228 // @event movestart: Event 4229 // Fired when the view of the map starts changing (e.g. user starts dragging the map). 4230 if (zoomChanged) { 4231 this.fire('zoomstart'); 4232 } 4233 if (!noMoveStart) { 4234 this.fire('movestart'); 4235 } 4236 return this; 4237 }, 4238 4239 _move: function (center, zoom, data) { 4240 if (zoom === undefined) { 4241 zoom = this._zoom; 4242 } 4243 var zoomChanged = this._zoom !== zoom; 4244 4245 this._zoom = zoom; 4246 this._lastCenter = center; 4247 this._pixelOrigin = this._getNewPixelOrigin(center); 4248 4249 // @event zoom: Event 4250 // Fired repeatedly during any change in zoom level, including zoom 4251 // and fly animations. 4252 if (zoomChanged || (data && data.pinch)) { // Always fire 'zoom' if pinching because #3530 4253 this.fire('zoom', data); 4254 } 4255 4256 // @event move: Event 4257 // Fired repeatedly during any movement of the map, including pan and 4258 // fly animations. 4259 return this.fire('move', data); 4260 }, 4261 4262 _moveEnd: function (zoomChanged) { 4263 // @event zoomend: Event 4264 // Fired when the map has changed, after any animations. 4265 if (zoomChanged) { 4266 this.fire('zoomend'); 4267 } 4268 4269 // @event moveend: Event 4270 // Fired when the center of the map stops changing (e.g. user stopped 4271 // dragging the map). 4272 return this.fire('moveend'); 4273 }, 4274 4275 _stop: function () { 4276 cancelAnimFrame(this._flyToFrame); 4277 if (this._panAnim) { 4278 this._panAnim.stop(); 4279 } 4280 return this; 4281 }, 4282 4283 _rawPanBy: function (offset) { 4284 setPosition(this._mapPane, this._getMapPanePos().subtract(offset)); 4285 }, 4286 4287 _getZoomSpan: function () { 4288 return this.getMaxZoom() - this.getMinZoom(); 4289 }, 4290 4291 _panInsideMaxBounds: function () { 4292 if (!this._enforcingBounds) { 4293 this.panInsideBounds(this.options.maxBounds); 4294 } 4295 }, 4296 4297 _checkIfLoaded: function () { 4298 if (!this._loaded) { 4299 throw new Error('Set map center and zoom first.'); 4300 } 4301 }, 4302 4303 // DOM event handling 4304 4305 // @section Interaction events 4306 _initEvents: function (remove$$1) { 4307 this._targets = {}; 4308 this._targets[stamp(this._container)] = this; 4309 4310 var onOff = remove$$1 ? off : on; 4311 4312 // @event click: MouseEvent 4313 // Fired when the user clicks (or taps) the map. 4314 // @event dblclick: MouseEvent 4315 // Fired when the user double-clicks (or double-taps) the map. 4316 // @event mousedown: MouseEvent 4317 // Fired when the user pushes the mouse button on the map. 4318 // @event mouseup: MouseEvent 4319 // Fired when the user releases the mouse button on the map. 4320 // @event mouseover: MouseEvent 4321 // Fired when the mouse enters the map. 4322 // @event mouseout: MouseEvent 4323 // Fired when the mouse leaves the map. 4324 // @event mousemove: MouseEvent 4325 // Fired while the mouse moves over the map. 4326 // @event contextmenu: MouseEvent 4327 // Fired when the user pushes the right mouse button on the map, prevents 4328 // default browser context menu from showing if there are listeners on 4329 // this event. Also fired on mobile when the user holds a single touch 4330 // for a second (also called long press). 4331 // @event keypress: KeyboardEvent 4332 // Fired when the user presses a key from the keyboard that produces a character value while the map is focused. 4333 // @event keydown: KeyboardEvent 4334 // Fired when the user presses a key from the keyboard while the map is focused. Unlike the `keypress` event, 4335 // the `keydown` event is fired for keys that produce a character value and for keys 4336 // that do not produce a character value. 4337 // @event keyup: KeyboardEvent 4338 // Fired when the user releases a key from the keyboard while the map is focused. 4339 onOff(this._container, 'click dblclick mousedown mouseup ' + 4340 'mouseover mouseout mousemove contextmenu keypress keydown keyup', this._handleDOMEvent, this); 4341 4342 if (this.options.trackResize) { 4343 onOff(window, 'resize', this._onResize, this); 4344 } 4345 4346 if (any3d && this.options.transform3DLimit) { 4347 (remove$$1 ? this.off : this.on).call(this, 'moveend', this._onMoveEnd); 4348 } 4349 }, 4350 4351 _onResize: function () { 4352 cancelAnimFrame(this._resizeRequest); 4353 this._resizeRequest = requestAnimFrame( 4354 function () { this.invalidateSize({debounceMoveend: true}); }, this); 4355 }, 4356 4357 _onScroll: function () { 4358 this._container.scrollTop = 0; 4359 this._container.scrollLeft = 0; 4360 }, 4361 4362 _onMoveEnd: function () { 4363 var pos = this._getMapPanePos(); 4364 if (Math.max(Math.abs(pos.x), Math.abs(pos.y)) >= this.options.transform3DLimit) { 4365 // https://bugzilla.mozilla.org/show_bug.cgi?id=1203873 but Webkit also have 4366 // a pixel offset on very high values, see: http://jsfiddle.net/dg6r5hhb/ 4367 this._resetView(this.getCenter(), this.getZoom()); 4368 } 4369 }, 4370 4371 _findEventTargets: function (e, type) { 4372 var targets = [], 4373 target, 4374 isHover = type === 'mouseout' || type === 'mouseover', 4375 src = e.target || e.srcElement, 4376 dragging = false; 4377 4378 while (src) { 4379 target = this._targets[stamp(src)]; 4380 if (target && (type === 'click' || type === 'preclick') && !e._simulated && this._draggableMoved(target)) { 4381 // Prevent firing click after you just dragged an object. 4382 dragging = true; 4383 break; 4384 } 4385 if (target && target.listens(type, true)) { 4386 if (isHover && !isExternalTarget(src, e)) { break; } 4387 targets.push(target); 4388 if (isHover) { break; } 4389 } 4390 if (src === this._container) { break; } 4391 src = src.parentNode; 4392 } 4393 if (!targets.length && !dragging && !isHover && isExternalTarget(src, e)) { 4394 targets = [this]; 4395 } 4396 return targets; 4397 }, 4398 4399 _handleDOMEvent: function (e) { 4400 if (!this._loaded || skipped(e)) { return; } 4401 4402 var type = e.type; 4403 4404 if (type === 'mousedown' || type === 'keypress' || type === 'keyup' || type === 'keydown') { 4405 // prevents outline when clicking on keyboard-focusable element 4406 preventOutline(e.target || e.srcElement); 4407 } 4408 4409 this._fireDOMEvent(e, type); 4410 }, 4411 4412 _mouseEvents: ['click', 'dblclick', 'mouseover', 'mouseout', 'contextmenu'], 4413 4414 _fireDOMEvent: function (e, type, targets) { 4415 4416 if (e.type === 'click') { 4417 // Fire a synthetic 'preclick' event which propagates up (mainly for closing popups). 4418 // @event preclick: MouseEvent 4419 // Fired before mouse click on the map (sometimes useful when you 4420 // want something to happen on click before any existing click 4421 // handlers start running). 4422 var synth = extend({}, e); 4423 synth.type = 'preclick'; 4424 this._fireDOMEvent(synth, synth.type, targets); 4425 } 4426 4427 if (e._stopped) { return; } 4428 4429 // Find the layer the event is propagating from and its parents. 4430 targets = (targets || []).concat(this._findEventTargets(e, type)); 4431 4432 if (!targets.length) { return; } 4433 4434 var target = targets[0]; 4435 if (type === 'contextmenu' && target.listens(type, true)) { 4436 preventDefault(e); 4437 } 4438 4439 var data = { 4440 originalEvent: e 4441 }; 4442 4443 if (e.type !== 'keypress' && e.type !== 'keydown' && e.type !== 'keyup') { 4444 var isMarker = target.getLatLng && (!target._radius || target._radius <= 10); 4445 data.containerPoint = isMarker ? 4446 this.latLngToContainerPoint(target.getLatLng()) : this.mouseEventToContainerPoint(e); 4447 data.layerPoint = this.containerPointToLayerPoint(data.containerPoint); 4448 data.latlng = isMarker ? target.getLatLng() : this.layerPointToLatLng(data.layerPoint); 4449 } 4450 4451 for (var i = 0; i < targets.length; i++) { 4452 targets[i].fire(type, data, true); 4453 if (data.originalEvent._stopped || 4454 (targets[i].options.bubblingMouseEvents === false && indexOf(this._mouseEvents, type) !== -1)) { return; } 4455 } 4456 }, 4457 4458 _draggableMoved: function (obj) { 4459 obj = obj.dragging && obj.dragging.enabled() ? obj : this; 4460 return (obj.dragging && obj.dragging.moved()) || (this.boxZoom && this.boxZoom.moved()); 4461 }, 4462 4463 _clearHandlers: function () { 4464 for (var i = 0, len = this._handlers.length; i < len; i++) { 4465 this._handlers[i].disable(); 4466 } 4467 }, 4468 4469 // @section Other Methods 4470 4471 // @method whenReady(fn: Function, context?: Object): this 4472 // Runs the given function `fn` when the map gets initialized with 4473 // a view (center and zoom) and at least one layer, or immediately 4474 // if it's already initialized, optionally passing a function context. 4475 whenReady: function (callback, context) { 4476 if (this._loaded) { 4477 callback.call(context || this, {target: this}); 4478 } else { 4479 this.on('load', callback, context); 4480 } 4481 return this; 4482 }, 4483 4484 4485 // private methods for getting map state 4486 4487 _getMapPanePos: function () { 4488 return getPosition(this._mapPane) || new Point(0, 0); 4489 }, 4490 4491 _moved: function () { 4492 var pos = this._getMapPanePos(); 4493 return pos && !pos.equals([0, 0]); 4494 }, 4495 4496 _getTopLeftPoint: function (center, zoom) { 4497 var pixelOrigin = center && zoom !== undefined ? 4498 this._getNewPixelOrigin(center, zoom) : 4499 this.getPixelOrigin(); 4500 return pixelOrigin.subtract(this._getMapPanePos()); 4501 }, 4502 4503 _getNewPixelOrigin: function (center, zoom) { 4504 var viewHalf = this.getSize()._divideBy(2); 4505 return this.project(center, zoom)._subtract(viewHalf)._add(this._getMapPanePos())._round(); 4506 }, 4507 4508 _latLngToNewLayerPoint: function (latlng, zoom, center) { 4509 var topLeft = this._getNewPixelOrigin(center, zoom); 4510 return this.project(latlng, zoom)._subtract(topLeft); 4511 }, 4512 4513 _latLngBoundsToNewLayerBounds: function (latLngBounds, zoom, center) { 4514 var topLeft = this._getNewPixelOrigin(center, zoom); 4515 return toBounds([ 4516 this.project(latLngBounds.getSouthWest(), zoom)._subtract(topLeft), 4517 this.project(latLngBounds.getNorthWest(), zoom)._subtract(topLeft), 4518 this.project(latLngBounds.getSouthEast(), zoom)._subtract(topLeft), 4519 this.project(latLngBounds.getNorthEast(), zoom)._subtract(topLeft) 4520 ]); 4521 }, 4522 4523 // layer point of the current center 4524 _getCenterLayerPoint: function () { 4525 return this.containerPointToLayerPoint(this.getSize()._divideBy(2)); 4526 }, 4527 4528 // offset of the specified place to the current center in pixels 4529 _getCenterOffset: function (latlng) { 4530 return this.latLngToLayerPoint(latlng).subtract(this._getCenterLayerPoint()); 4531 }, 4532 4533 // adjust center for view to get inside bounds 4534 _limitCenter: function (center, zoom, bounds) { 4535 4536 if (!bounds) { return center; } 4537 4538 var centerPoint = this.project(center, zoom), 4539 viewHalf = this.getSize().divideBy(2), 4540 viewBounds = new Bounds(centerPoint.subtract(viewHalf), centerPoint.add(viewHalf)), 4541 offset = this._getBoundsOffset(viewBounds, bounds, zoom); 4542 4543 // If offset is less than a pixel, ignore. 4544 // This prevents unstable projections from getting into 4545 // an infinite loop of tiny offsets. 4546 if (offset.round().equals([0, 0])) { 4547 return center; 4548 } 4549 4550 return this.unproject(centerPoint.add(offset), zoom); 4551 }, 4552 4553 // adjust offset for view to get inside bounds 4554 _limitOffset: function (offset, bounds) { 4555 if (!bounds) { return offset; } 4556 4557 var viewBounds = this.getPixelBounds(), 4558 newBounds = new Bounds(viewBounds.min.add(offset), viewBounds.max.add(offset)); 4559 4560 return offset.add(this._getBoundsOffset(newBounds, bounds)); 4561 }, 4562 4563 // returns offset needed for pxBounds to get inside maxBounds at a specified zoom 4564 _getBoundsOffset: function (pxBounds, maxBounds, zoom) { 4565 var projectedMaxBounds = toBounds( 4566 this.project(maxBounds.getNorthEast(), zoom), 4567 this.project(maxBounds.getSouthWest(), zoom) 4568 ), 4569 minOffset = projectedMaxBounds.min.subtract(pxBounds.min), 4570 maxOffset = projectedMaxBounds.max.subtract(pxBounds.max), 4571 4572 dx = this._rebound(minOffset.x, -maxOffset.x), 4573 dy = this._rebound(minOffset.y, -maxOffset.y); 4574 4575 return new Point(dx, dy); 4576 }, 4577 4578 _rebound: function (left, right) { 4579 return left + right > 0 ? 4580 Math.round(left - right) / 2 : 4581 Math.max(0, Math.ceil(left)) - Math.max(0, Math.floor(right)); 4582 }, 4583 4584 _limitZoom: function (zoom) { 4585 var min = this.getMinZoom(), 4586 max = this.getMaxZoom(), 4587 snap = any3d ? this.options.zoomSnap : 1; 4588 if (snap) { 4589 zoom = Math.round(zoom / snap) * snap; 4590 } 4591 return Math.max(min, Math.min(max, zoom)); 4592 }, 4593 4594 _onPanTransitionStep: function () { 4595 this.fire('move'); 4596 }, 4597 4598 _onPanTransitionEnd: function () { 4599 removeClass(this._mapPane, 'leaflet-pan-anim'); 4600 this.fire('moveend'); 4601 }, 4602 4603 _tryAnimatedPan: function (center, options) { 4604 // difference between the new and current centers in pixels 4605 var offset = this._getCenterOffset(center)._trunc(); 4606 4607 // don't animate too far unless animate: true specified in options 4608 if ((options && options.animate) !== true && !this.getSize().contains(offset)) { return false; } 4609 4610 this.panBy(offset, options); 4611 4612 return true; 4613 }, 4614 4615 _createAnimProxy: function () { 4616 4617 var proxy = this._proxy = create$1('div', 'leaflet-proxy leaflet-zoom-animated'); 4618 this._panes.mapPane.appendChild(proxy); 4619 4620 this.on('zoomanim', function (e) { 4621 var prop = TRANSFORM, 4622 transform = this._proxy.style[prop]; 4623 4624 setTransform(this._proxy, this.project(e.center, e.zoom), this.getZoomScale(e.zoom, 1)); 4625 4626 // workaround for case when transform is the same and so transitionend event is not fired 4627 if (transform === this._proxy.style[prop] && this._animatingZoom) { 4628 this._onZoomTransitionEnd(); 4629 } 4630 }, this); 4631 4632 this.on('load moveend', this._animMoveEnd, this); 4633 4634 this._on('unload', this._destroyAnimProxy, this); 4635 }, 4636 4637 _destroyAnimProxy: function () { 4638 remove(this._proxy); 4639 this.off('load moveend', this._animMoveEnd, this); 4640 delete this._proxy; 4641 }, 4642 4643 _animMoveEnd: function () { 4644 var c = this.getCenter(), 4645 z = this.getZoom(); 4646 setTransform(this._proxy, this.project(c, z), this.getZoomScale(z, 1)); 4647 }, 4648 4649 _catchTransitionEnd: function (e) { 4650 if (this._animatingZoom && e.propertyName.indexOf('transform') >= 0) { 4651 this._onZoomTransitionEnd(); 4652 } 4653 }, 4654 4655 _nothingToAnimate: function () { 4656 return !this._container.getElementsByClassName('leaflet-zoom-animated').length; 4657 }, 4658 4659 _tryAnimatedZoom: function (center, zoom, options) { 4660 4661 if (this._animatingZoom) { return true; } 4662 4663 options = options || {}; 4664 4665 // don't animate if disabled, not supported or zoom difference is too large 4666 if (!this._zoomAnimated || options.animate === false || this._nothingToAnimate() || 4667 Math.abs(zoom - this._zoom) > this.options.zoomAnimationThreshold) { return false; } 4668 4669 // offset is the pixel coords of the zoom origin relative to the current center 4670 var scale = this.getZoomScale(zoom), 4671 offset = this._getCenterOffset(center)._divideBy(1 - 1 / scale); 4672 4673 // don't animate if the zoom origin isn't within one screen from the current center, unless forced 4674 if (options.animate !== true && !this.getSize().contains(offset)) { return false; } 4675 4676 requestAnimFrame(function () { 4677 this 4678 ._moveStart(true, false) 4679 ._animateZoom(center, zoom, true); 4680 }, this); 4681 4682 return true; 4683 }, 4684 4685 _animateZoom: function (center, zoom, startAnim, noUpdate) { 4686 if (!this._mapPane) { return; } 4687 4688 if (startAnim) { 4689 this._animatingZoom = true; 4690 4691 // remember what center/zoom to set after animation 4692 this._animateToCenter = center; 4693 this._animateToZoom = zoom; 4694 4695 addClass(this._mapPane, 'leaflet-zoom-anim'); 4696 } 4697 4698 // @section Other Events 4699 // @event zoomanim: ZoomAnimEvent 4700 // Fired at least once per zoom animation. For continuous zoom, like pinch zooming, fired once per frame during zoom. 4701 this.fire('zoomanim', { 4702 center: center, 4703 zoom: zoom, 4704 noUpdate: noUpdate 4705 }); 4706 4707 // Work around webkit not firing 'transitionend', see https://github.com/Leaflet/Leaflet/issues/3689, 2693 4708 setTimeout(bind(this._onZoomTransitionEnd, this), 250); 4709 }, 4710 4711 _onZoomTransitionEnd: function () { 4712 if (!this._animatingZoom) { return; } 4713 4714 if (this._mapPane) { 4715 removeClass(this._mapPane, 'leaflet-zoom-anim'); 4716 } 4717 4718 this._animatingZoom = false; 4719 4720 this._move(this._animateToCenter, this._animateToZoom); 4721 4722 // This anim frame should prevent an obscure iOS webkit tile loading race condition. 4723 requestAnimFrame(function () { 4724 this._moveEnd(true); 4725 }, this); 4726 } 4727 }); 4728 4729 // @section 4730 4731 // @factory L.map(id: String, options?: Map options) 4732 // Instantiates a map object given the DOM ID of a `<div>` element 4733 // and optionally an object literal with `Map options`. 4734 // 4735 // @alternative 4736 // @factory L.map(el: HTMLElement, options?: Map options) 4737 // Instantiates a map object given an instance of a `<div>` HTML element 4738 // and optionally an object literal with `Map options`. 4739 function createMap(id, options) { 4740 return new Map(id, options); 4741 } 4742 4743 /* 4744 * @class Control 4745 * @aka L.Control 4746 * @inherits Class 4747 * 4748 * L.Control is a base class for implementing map controls. Handles positioning. 4749 * All other controls extend from this class. 4750 */ 4751 4752 var Control = Class.extend({ 4753 // @section 4754 // @aka Control options 4755 options: { 4756 // @option position: String = 'topright' 4757 // The position of the control (one of the map corners). Possible values are `'topleft'`, 4758 // `'topright'`, `'bottomleft'` or `'bottomright'` 4759 position: 'topright' 4760 }, 4761 4762 initialize: function (options) { 4763 setOptions(this, options); 4764 }, 4765 4766 /* @section 4767 * Classes extending L.Control will inherit the following methods: 4768 * 4769 * @method getPosition: string 4770 * Returns the position of the control. 4771 */ 4772 getPosition: function () { 4773 return this.options.position; 4774 }, 4775 4776 // @method setPosition(position: string): this 4777 // Sets the position of the control. 4778 setPosition: function (position) { 4779 var map = this._map; 4780 4781 if (map) { 4782 map.removeControl(this); 4783 } 4784 4785 this.options.position = position; 4786 4787 if (map) { 4788 map.addControl(this); 4789 } 4790 4791 return this; 4792 }, 4793 4794 // @method getContainer: HTMLElement 4795 // Returns the HTMLElement that contains the control. 4796 getContainer: function () { 4797 return this._container; 4798 }, 4799 4800 // @method addTo(map: Map): this 4801 // Adds the control to the given map. 4802 addTo: function (map) { 4803 this.remove(); 4804 this._map = map; 4805 4806 var container = this._container = this.onAdd(map), 4807 pos = this.getPosition(), 4808 corner = map._controlCorners[pos]; 4809 4810 addClass(container, 'leaflet-control'); 4811 4812 if (pos.indexOf('bottom') !== -1) { 4813 corner.insertBefore(container, corner.firstChild); 4814 } else { 4815 corner.appendChild(container); 4816 } 4817 4818 this._map.on('unload', this.remove, this); 4819 4820 return this; 4821 }, 4822 4823 // @method remove: this 4824 // Removes the control from the map it is currently active on. 4825 remove: function () { 4826 if (!this._map) { 4827 return this; 4828 } 4829 4830 remove(this._container); 4831 4832 if (this.onRemove) { 4833 this.onRemove(this._map); 4834 } 4835 4836 this._map.off('unload', this.remove, this); 4837 this._map = null; 4838 4839 return this; 4840 }, 4841 4842 _refocusOnMap: function (e) { 4843 // if map exists and event is not a keyboard event 4844 if (this._map && e && e.screenX > 0 && e.screenY > 0) { 4845 this._map.getContainer().focus(); 4846 } 4847 } 4848 }); 4849 4850 var control = function (options) { 4851 return new Control(options); 4852 }; 4853 4854 /* @section Extension methods 4855 * @uninheritable 4856 * 4857 * Every control should extend from `L.Control` and (re-)implement the following methods. 4858 * 4859 * @method onAdd(map: Map): HTMLElement 4860 * Should return the container DOM element for the control and add listeners on relevant map events. Called on [`control.addTo(map)`](#control-addTo). 4861 * 4862 * @method onRemove(map: Map) 4863 * Optional method. Should contain all clean up code that removes the listeners previously added in [`onAdd`](#control-onadd). Called on [`control.remove()`](#control-remove). 4864 */ 4865 4866 /* @namespace Map 4867 * @section Methods for Layers and Controls 4868 */ 4869 Map.include({ 4870 // @method addControl(control: Control): this 4871 // Adds the given control to the map 4872 addControl: function (control) { 4873 control.addTo(this); 4874 return this; 4875 }, 4876 4877 // @method removeControl(control: Control): this 4878 // Removes the given control from the map 4879 removeControl: function (control) { 4880 control.remove(); 4881 return this; 4882 }, 4883 4884 _initControlPos: function () { 4885 var corners = this._controlCorners = {}, 4886 l = 'leaflet-', 4887 container = this._controlContainer = 4888 create$1('div', l + 'control-container', this._container); 4889 4890 function createCorner(vSide, hSide) { 4891 var className = l + vSide + ' ' + l + hSide; 4892 4893 corners[vSide + hSide] = create$1('div', className, container); 4894 } 4895 4896 createCorner('top', 'left'); 4897 createCorner('top', 'right'); 4898 createCorner('bottom', 'left'); 4899 createCorner('bottom', 'right'); 4900 }, 4901 4902 _clearControlPos: function () { 4903 for (var i in this._controlCorners) { 4904 remove(this._controlCorners[i]); 4905 } 4906 remove(this._controlContainer); 4907 delete this._controlCorners; 4908 delete this._controlContainer; 4909 } 4910 }); 4911 4912 /* 4913 * @class Control.Layers 4914 * @aka L.Control.Layers 4915 * @inherits Control 4916 * 4917 * The layers control gives users the ability to switch between different base layers and switch overlays on/off (check out the [detailed example](http://leafletjs.com/examples/layers-control/)). Extends `Control`. 4918 * 4919 * @example 4920 * 4921 * ```js 4922 * var baseLayers = { 4923 * "Mapbox": mapbox, 4924 * "OpenStreetMap": osm 4925 * }; 4926 * 4927 * var overlays = { 4928 * "Marker": marker, 4929 * "Roads": roadsLayer 4930 * }; 4931 * 4932 * L.control.layers(baseLayers, overlays).addTo(map); 4933 * ``` 4934 * 4935 * The `baseLayers` and `overlays` parameters are object literals with layer names as keys and `Layer` objects as values: 4936 * 4937 * ```js 4938 * { 4939 * "<someName1>": layer1, 4940 * "<someName2>": layer2 4941 * } 4942 * ``` 4943 * 4944 * The layer names can contain HTML, which allows you to add additional styling to the items: 4945 * 4946 * ```js 4947 * {"<img src='my-layer-icon' /> <span class='my-layer-item'>My Layer</span>": myLayer} 4948 * ``` 4949 */ 4950 4951 var Layers = Control.extend({ 4952 // @section 4953 // @aka Control.Layers options 4954 options: { 4955 // @option collapsed: Boolean = true 4956 // If `true`, the control will be collapsed into an icon and expanded on mouse hover or touch. 4957 collapsed: true, 4958 position: 'topright', 4959 4960 // @option autoZIndex: Boolean = true 4961 // If `true`, the control will assign zIndexes in increasing order to all of its layers so that the order is preserved when switching them on/off. 4962 autoZIndex: true, 4963 4964 // @option hideSingleBase: Boolean = false 4965 // If `true`, the base layers in the control will be hidden when there is only one. 4966 hideSingleBase: false, 4967 4968 // @option sortLayers: Boolean = false 4969 // Whether to sort the layers. When `false`, layers will keep the order 4970 // in which they were added to the control. 4971 sortLayers: false, 4972 4973 // @option sortFunction: Function = * 4974 // A [compare function](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Array/sort) 4975 // that will be used for sorting the layers, when `sortLayers` is `true`. 4976 // The function receives both the `L.Layer` instances and their names, as in 4977 // `sortFunction(layerA, layerB, nameA, nameB)`. 4978 // By default, it sorts layers alphabetically by their name. 4979 sortFunction: function (layerA, layerB, nameA, nameB) { 4980 return nameA < nameB ? -1 : (nameB < nameA ? 1 : 0); 4981 } 4982 }, 4983 4984 initialize: function (baseLayers, overlays, options) { 4985 setOptions(this, options); 4986 4987 this._layerControlInputs = []; 4988 this._layers = []; 4989 this._lastZIndex = 0; 4990 this._handlingClick = false; 4991 4992 for (var i in baseLayers) { 4993 this._addLayer(baseLayers[i], i); 4994 } 4995 4996 for (i in overlays) { 4997 this._addLayer(overlays[i], i, true); 4998 } 4999 }, 5000 5001 onAdd: function (map) { 5002 this._initLayout(); 5003 this._update(); 5004 5005 this._map = map; 5006 map.on('zoomend', this._checkDisabledLayers, this); 5007 5008 for (var i = 0; i < this._layers.length; i++) { 5009 this._layers[i].layer.on('add remove', this._onLayerChange, this); 5010 } 5011 5012 return this._container; 5013 }, 5014 5015 addTo: function (map) { 5016 Control.prototype.addTo.call(this, map); 5017 // Trigger expand after Layers Control has been inserted into DOM so that is now has an actual height. 5018 return this._expandIfNotCollapsed(); 5019 }, 5020 5021 onRemove: function () { 5022 this._map.off('zoomend', this._checkDisabledLayers, this); 5023 5024 for (var i = 0; i < this._layers.length; i++) { 5025 this._layers[i].layer.off('add remove', this._onLayerChange, this); 5026 } 5027 }, 5028 5029 // @method addBaseLayer(layer: Layer, name: String): this 5030 // Adds a base layer (radio button entry) with the given name to the control. 5031 addBaseLayer: function (layer, name) { 5032 this._addLayer(layer, name); 5033 return (this._map) ? this._update() : this; 5034 }, 5035 5036 // @method addOverlay(layer: Layer, name: String): this 5037 // Adds an overlay (checkbox entry) with the given name to the control. 5038 addOverlay: function (layer, name) { 5039 this._addLayer(layer, name, true); 5040 return (this._map) ? this._update() : this; 5041 }, 5042 5043 // @method removeLayer(layer: Layer): this 5044 // Remove the given layer from the control. 5045 removeLayer: function (layer) { 5046 layer.off('add remove', this._onLayerChange, this); 5047 5048 var obj = this._getLayer(stamp(layer)); 5049 if (obj) { 5050 this._layers.splice(this._layers.indexOf(obj), 1); 5051 } 5052 return (this._map) ? this._update() : this; 5053 }, 5054 5055 // @method expand(): this 5056 // Expand the control container if collapsed. 5057 expand: function () { 5058 addClass(this._container, 'leaflet-control-layers-expanded'); 5059 this._section.style.height = null; 5060 var acceptableHeight = this._map.getSize().y - (this._container.offsetTop + 50); 5061 if (acceptableHeight < this._section.clientHeight) { 5062 addClass(this._section, 'leaflet-control-layers-scrollbar'); 5063 this._section.style.height = acceptableHeight + 'px'; 5064 } else { 5065 removeClass(this._section, 'leaflet-control-layers-scrollbar'); 5066 } 5067 this._checkDisabledLayers(); 5068 return this; 5069 }, 5070 5071 // @method collapse(): this 5072 // Collapse the control container if expanded. 5073 collapse: function () { 5074 removeClass(this._container, 'leaflet-control-layers-expanded'); 5075 return this; 5076 }, 5077 5078 _initLayout: function () { 5079 var className = 'leaflet-control-layers', 5080 container = this._container = create$1('div', className), 5081 collapsed = this.options.collapsed; 5082 5083 // makes this work on IE touch devices by stopping it from firing a mouseout event when the touch is released 5084 container.setAttribute('aria-haspopup', true); 5085 5086 disableClickPropagation(container); 5087 disableScrollPropagation(container); 5088 5089 var section = this._section = create$1('section', className + '-list'); 5090 5091 if (collapsed) { 5092 this._map.on('click', this.collapse, this); 5093 5094 if (!android) { 5095 on(container, { 5096 mouseenter: this.expand, 5097 mouseleave: this.collapse 5098 }, this); 5099 } 5100 } 5101 5102 var link = this._layersLink = create$1('a', className + '-toggle', container); 5103 link.href = '#'; 5104 link.title = 'Layers'; 5105 5106 if (touch) { 5107 on(link, 'click', stop); 5108 on(link, 'click', this.expand, this); 5109 } else { 5110 on(link, 'focus', this.expand, this); 5111 } 5112 5113 if (!collapsed) { 5114 this.expand(); 5115 } 5116 5117 this._baseLayersList = create$1('div', className + '-base', section); 5118 this._separator = create$1('div', className + '-separator', section); 5119 this._overlaysList = create$1('div', className + '-overlays', section); 5120 5121 container.appendChild(section); 5122 }, 5123 5124 _getLayer: function (id) { 5125 for (var i = 0; i < this._layers.length; i++) { 5126 5127 if (this._layers[i] && stamp(this._layers[i].layer) === id) { 5128 return this._layers[i]; 5129 } 5130 } 5131 }, 5132 5133 _addLayer: function (layer, name, overlay) { 5134 if (this._map) { 5135 layer.on('add remove', this._onLayerChange, this); 5136 } 5137 5138 this._layers.push({ 5139 layer: layer, 5140 name: name, 5141 overlay: overlay 5142 }); 5143 5144 if (this.options.sortLayers) { 5145 this._layers.sort(bind(function (a, b) { 5146 return this.options.sortFunction(a.layer, b.layer, a.name, b.name); 5147 }, this)); 5148 } 5149 5150 if (this.options.autoZIndex && layer.setZIndex) { 5151 this._lastZIndex++; 5152 layer.setZIndex(this._lastZIndex); 5153 } 5154 5155 this._expandIfNotCollapsed(); 5156 }, 5157 5158 _update: function () { 5159 if (!this._container) { return this; } 5160 5161 empty(this._baseLayersList); 5162 empty(this._overlaysList); 5163 5164 this._layerControlInputs = []; 5165 var baseLayersPresent, overlaysPresent, i, obj, baseLayersCount = 0; 5166 5167 for (i = 0; i < this._layers.length; i++) { 5168 obj = this._layers[i]; 5169 this._addItem(obj); 5170 overlaysPresent = overlaysPresent || obj.overlay; 5171 baseLayersPresent = baseLayersPresent || !obj.overlay; 5172 baseLayersCount += !obj.overlay ? 1 : 0; 5173 } 5174 5175 // Hide base layers section if there's only one layer. 5176 if (this.options.hideSingleBase) { 5177 baseLayersPresent = baseLayersPresent && baseLayersCount > 1; 5178 this._baseLayersList.style.display = baseLayersPresent ? '' : 'none'; 5179 } 5180 5181 this._separator.style.display = overlaysPresent && baseLayersPresent ? '' : 'none'; 5182 5183 return this; 5184 }, 5185 5186 _onLayerChange: function (e) { 5187 if (!this._handlingClick) { 5188 this._update(); 5189 } 5190 5191 var obj = this._getLayer(stamp(e.target)); 5192 5193 // @namespace Map 5194 // @section Layer events 5195 // @event baselayerchange: LayersControlEvent 5196 // Fired when the base layer is changed through the [layer control](#control-layers). 5197 // @event overlayadd: LayersControlEvent 5198 // Fired when an overlay is selected through the [layer control](#control-layers). 5199 // @event overlayremove: LayersControlEvent 5200 // Fired when an overlay is deselected through the [layer control](#control-layers). 5201 // @namespace Control.Layers 5202 var type = obj.overlay ? 5203 (e.type === 'add' ? 'overlayadd' : 'overlayremove') : 5204 (e.type === 'add' ? 'baselayerchange' : null); 5205 5206 if (type) { 5207 this._map.fire(type, obj); 5208 } 5209 }, 5210 5211 // IE7 bugs out if you create a radio dynamically, so you have to do it this hacky way (see http://bit.ly/PqYLBe) 5212 _createRadioElement: function (name, checked) { 5213 5214 var radioHtml = '<input type="radio" class="leaflet-control-layers-selector" name="' + 5215 name + '"' + (checked ? ' checked="checked"' : '') + '/>'; 5216 5217 var radioFragment = document.createElement('div'); 5218 radioFragment.innerHTML = radioHtml; 5219 5220 return radioFragment.firstChild; 5221 }, 5222 5223 _addItem: function (obj) { 5224 var label = document.createElement('label'), 5225 checked = this._map.hasLayer(obj.layer), 5226 input; 5227 5228 if (obj.overlay) { 5229 input = document.createElement('input'); 5230 input.type = 'checkbox'; 5231 input.className = 'leaflet-control-layers-selector'; 5232 input.defaultChecked = checked; 5233 } else { 5234 input = this._createRadioElement('leaflet-base-layers_' + stamp(this), checked); 5235 } 5236 5237 this._layerControlInputs.push(input); 5238 input.layerId = stamp(obj.layer); 5239 5240 on(input, 'click', this._onInputClick, this); 5241 5242 var name = document.createElement('span'); 5243 name.innerHTML = ' ' + obj.name; 5244 5245 // Helps from preventing layer control flicker when checkboxes are disabled 5246 // https://github.com/Leaflet/Leaflet/issues/2771 5247 var holder = document.createElement('div'); 5248 5249 label.appendChild(holder); 5250 holder.appendChild(input); 5251 holder.appendChild(name); 5252 5253 var container = obj.overlay ? this._overlaysList : this._baseLayersList; 5254 container.appendChild(label); 5255 5256 this._checkDisabledLayers(); 5257 return label; 5258 }, 5259 5260 _onInputClick: function () { 5261 var inputs = this._layerControlInputs, 5262 input, layer; 5263 var addedLayers = [], 5264 removedLayers = []; 5265 5266 this._handlingClick = true; 5267 5268 for (var i = inputs.length - 1; i >= 0; i--) { 5269 input = inputs[i]; 5270 layer = this._getLayer(input.layerId).layer; 5271 5272 if (input.checked) { 5273 addedLayers.push(layer); 5274 } else if (!input.checked) { 5275 removedLayers.push(layer); 5276 } 5277 } 5278 5279 // Bugfix issue 2318: Should remove all old layers before readding new ones 5280 for (i = 0; i < removedLayers.length; i++) { 5281 if (this._map.hasLayer(removedLayers[i])) { 5282 this._map.removeLayer(removedLayers[i]); 5283 } 5284 } 5285 for (i = 0; i < addedLayers.length; i++) { 5286 if (!this._map.hasLayer(addedLayers[i])) { 5287 this._map.addLayer(addedLayers[i]); 5288 } 5289 } 5290 5291 this._handlingClick = false; 5292 5293 this._refocusOnMap(); 5294 }, 5295 5296 _checkDisabledLayers: function () { 5297 var inputs = this._layerControlInputs, 5298 input, 5299 layer, 5300 zoom = this._map.getZoom(); 5301 5302 for (var i = inputs.length - 1; i >= 0; i--) { 5303 input = inputs[i]; 5304 layer = this._getLayer(input.layerId).layer; 5305 input.disabled = (layer.options.minZoom !== undefined && zoom < layer.options.minZoom) || 5306 (layer.options.maxZoom !== undefined && zoom > layer.options.maxZoom); 5307 5308 } 5309 }, 5310 5311 _expandIfNotCollapsed: function () { 5312 if (this._map && !this.options.collapsed) { 5313 this.expand(); 5314 } 5315 return this; 5316 }, 5317 5318 _expand: function () { 5319 // Backward compatibility, remove me in 1.1. 5320 return this.expand(); 5321 }, 5322 5323 _collapse: function () { 5324 // Backward compatibility, remove me in 1.1. 5325 return this.collapse(); 5326 } 5327 5328 }); 5329 5330 5331 // @factory L.control.layers(baselayers?: Object, overlays?: Object, options?: Control.Layers options) 5332 // Creates a layers control with the given layers. Base layers will be switched with radio buttons, while overlays will be switched with checkboxes. Note that all base layers should be passed in the base layers object, but only one should be added to the map during map instantiation. 5333 var layers = function (baseLayers, overlays, options) { 5334 return new Layers(baseLayers, overlays, options); 5335 }; 5336 5337 /* 5338 * @class Control.Zoom 5339 * @aka L.Control.Zoom 5340 * @inherits Control 5341 * 5342 * A basic zoom control with two buttons (zoom in and zoom out). It is put on the map by default unless you set its [`zoomControl` option](#map-zoomcontrol) to `false`. Extends `Control`. 5343 */ 5344 5345 var Zoom = Control.extend({ 5346 // @section 5347 // @aka Control.Zoom options 5348 options: { 5349 position: 'topleft', 5350 5351 // @option zoomInText: String = '+' 5352 // The text set on the 'zoom in' button. 5353 zoomInText: '+', 5354 5355 // @option zoomInTitle: String = 'Zoom in' 5356 // The title set on the 'zoom in' button. 5357 zoomInTitle: 'Zoom in', 5358 5359 // @option zoomOutText: String = '−' 5360 // The text set on the 'zoom out' button. 5361 zoomOutText: '−', 5362 5363 // @option zoomOutTitle: String = 'Zoom out' 5364 // The title set on the 'zoom out' button. 5365 zoomOutTitle: 'Zoom out' 5366 }, 5367 5368 onAdd: function (map) { 5369 var zoomName = 'leaflet-control-zoom', 5370 container = create$1('div', zoomName + ' leaflet-bar'), 5371 options = this.options; 5372 5373 this._zoomInButton = this._createButton(options.zoomInText, options.zoomInTitle, 5374 zoomName + '-in', container, this._zoomIn); 5375 this._zoomOutButton = this._createButton(options.zoomOutText, options.zoomOutTitle, 5376 zoomName + '-out', container, this._zoomOut); 5377 5378 this._updateDisabled(); 5379 map.on('zoomend zoomlevelschange', this._updateDisabled, this); 5380 5381 return container; 5382 }, 5383 5384 onRemove: function (map) { 5385 map.off('zoomend zoomlevelschange', this._updateDisabled, this); 5386 }, 5387 5388 disable: function () { 5389 this._disabled = true; 5390 this._updateDisabled(); 5391 return this; 5392 }, 5393 5394 enable: function () { 5395 this._disabled = false; 5396 this._updateDisabled(); 5397 return this; 5398 }, 5399 5400 _zoomIn: function (e) { 5401 if (!this._disabled && this._map._zoom < this._map.getMaxZoom()) { 5402 this._map.zoomIn(this._map.options.zoomDelta * (e.shiftKey ? 3 : 1)); 5403 } 5404 }, 5405 5406 _zoomOut: function (e) { 5407 if (!this._disabled && this._map._zoom > this._map.getMinZoom()) { 5408 this._map.zoomOut(this._map.options.zoomDelta * (e.shiftKey ? 3 : 1)); 5409 } 5410 }, 5411 5412 _createButton: function (html, title, className, container, fn) { 5413 var link = create$1('a', className, container); 5414 link.innerHTML = html; 5415 link.href = '#'; 5416 link.title = title; 5417 5418 /* 5419 * Will force screen readers like VoiceOver to read this as "Zoom in - button" 5420 */ 5421 link.setAttribute('role', 'button'); 5422 link.setAttribute('aria-label', title); 5423 5424 disableClickPropagation(link); 5425 on(link, 'click', stop); 5426 on(link, 'click', fn, this); 5427 on(link, 'click', this._refocusOnMap, this); 5428 5429 return link; 5430 }, 5431 5432 _updateDisabled: function () { 5433 var map = this._map, 5434 className = 'leaflet-disabled'; 5435 5436 removeClass(this._zoomInButton, className); 5437 removeClass(this._zoomOutButton, className); 5438 5439 if (this._disabled || map._zoom === map.getMinZoom()) { 5440 addClass(this._zoomOutButton, className); 5441 } 5442 if (this._disabled || map._zoom === map.getMaxZoom()) { 5443 addClass(this._zoomInButton, className); 5444 } 5445 } 5446 }); 5447 5448 // @namespace Map 5449 // @section Control options 5450 // @option zoomControl: Boolean = true 5451 // Whether a [zoom control](#control-zoom) is added to the map by default. 5452 Map.mergeOptions({ 5453 zoomControl: true 5454 }); 5455 5456 Map.addInitHook(function () { 5457 if (this.options.zoomControl) { 5458 // @section Controls 5459 // @property zoomControl: Control.Zoom 5460 // The default zoom control (only available if the 5461 // [`zoomControl` option](#map-zoomcontrol) was `true` when creating the map). 5462 this.zoomControl = new Zoom(); 5463 this.addControl(this.zoomControl); 5464 } 5465 }); 5466 5467 // @namespace Control.Zoom 5468 // @factory L.control.zoom(options: Control.Zoom options) 5469 // Creates a zoom control 5470 var zoom = function (options) { 5471 return new Zoom(options); 5472 }; 5473 5474 /* 5475 * @class Control.Scale 5476 * @aka L.Control.Scale 5477 * @inherits Control 5478 * 5479 * A simple scale control that shows the scale of the current center of screen in metric (m/km) and imperial (mi/ft) systems. Extends `Control`. 5480 * 5481 * @example 5482 * 5483 * ```js 5484 * L.control.scale().addTo(map); 5485 * ``` 5486 */ 5487 5488 var Scale = Control.extend({ 5489 // @section 5490 // @aka Control.Scale options 5491 options: { 5492 position: 'bottomleft', 5493 5494 // @option maxWidth: Number = 100 5495 // Maximum width of the control in pixels. The width is set dynamically to show round values (e.g. 100, 200, 500). 5496 maxWidth: 100, 5497 5498 // @option metric: Boolean = True 5499 // Whether to show the metric scale line (m/km). 5500 metric: true, 5501 5502 // @option imperial: Boolean = True 5503 // Whether to show the imperial scale line (mi/ft). 5504 imperial: true 5505 5506 // @option updateWhenIdle: Boolean = false 5507 // If `true`, the control is updated on [`moveend`](#map-moveend), otherwise it's always up-to-date (updated on [`move`](#map-move)). 5508 }, 5509 5510 onAdd: function (map) { 5511 var className = 'leaflet-control-scale', 5512 container = create$1('div', className), 5513 options = this.options; 5514 5515 this._addScales(options, className + '-line', container); 5516 5517 map.on(options.updateWhenIdle ? 'moveend' : 'move', this._update, this); 5518 map.whenReady(this._update, this); 5519 5520 return container; 5521 }, 5522 5523 onRemove: function (map) { 5524 map.off(this.options.updateWhenIdle ? 'moveend' : 'move', this._update, this); 5525 }, 5526 5527 _addScales: function (options, className, container) { 5528 if (options.metric) { 5529 this._mScale = create$1('div', className, container); 5530 } 5531 if (options.imperial) { 5532 this._iScale = create$1('div', className, container); 5533 } 5534 }, 5535 5536 _update: function () { 5537 var map = this._map, 5538 y = map.getSize().y / 2; 5539 5540 var maxMeters = map.distance( 5541 map.containerPointToLatLng([0, y]), 5542 map.containerPointToLatLng([this.options.maxWidth, y])); 5543 5544 this._updateScales(maxMeters); 5545 }, 5546 5547 _updateScales: function (maxMeters) { 5548 if (this.options.metric && maxMeters) { 5549 this._updateMetric(maxMeters); 5550 } 5551 if (this.options.imperial && maxMeters) { 5552 this._updateImperial(maxMeters); 5553 } 5554 }, 5555 5556 _updateMetric: function (maxMeters) { 5557 var meters = this._getRoundNum(maxMeters), 5558 label = meters < 1000 ? meters + ' m' : (meters / 1000) + ' km'; 5559 5560 this._updateScale(this._mScale, label, meters / maxMeters); 5561 }, 5562 5563 _updateImperial: function (maxMeters) { 5564 var maxFeet = maxMeters * 3.2808399, 5565 maxMiles, miles, feet; 5566 5567 if (maxFeet > 5280) { 5568 maxMiles = maxFeet / 5280; 5569 miles = this._getRoundNum(maxMiles); 5570 this._updateScale(this._iScale, miles + ' mi', miles / maxMiles); 5571 5572 } else { 5573 feet = this._getRoundNum(maxFeet); 5574 this._updateScale(this._iScale, feet + ' ft', feet / maxFeet); 5575 } 5576 }, 5577 5578 _updateScale: function (scale, text, ratio) { 5579 scale.style.width = Math.round(this.options.maxWidth * ratio) + 'px'; 5580 scale.innerHTML = text; 5581 }, 5582 5583 _getRoundNum: function (num) { 5584 var pow10 = Math.pow(10, (Math.floor(num) + '').length - 1), 5585 d = num / pow10; 5586 5587 d = d >= 10 ? 10 : 5588 d >= 5 ? 5 : 5589 d >= 3 ? 3 : 5590 d >= 2 ? 2 : 1; 5591 5592 return pow10 * d; 5593 } 5594 }); 5595 5596 5597 // @factory L.control.scale(options?: Control.Scale options) 5598 // Creates an scale control with the given options. 5599 var scale = function (options) { 5600 return new Scale(options); 5601 }; 5602 5603 /* 5604 * @class Control.Attribution 5605 * @aka L.Control.Attribution 5606 * @inherits Control 5607 * 5608 * The attribution control allows you to display attribution data in a small text box on a map. It is put on the map by default unless you set its [`attributionControl` option](#map-attributioncontrol) to `false`, and it fetches attribution texts from layers with the [`getAttribution` method](#layer-getattribution) automatically. Extends Control. 5609 */ 5610 5611 var Attribution = Control.extend({ 5612 // @section 5613 // @aka Control.Attribution options 5614 options: { 5615 position: 'bottomright', 5616 5617 // @option prefix: String = 'Leaflet' 5618 // The HTML text shown before the attributions. Pass `false` to disable. 5619 prefix: '<a href="https://leafletjs.com" title="A JS library for interactive maps">Leaflet</a>' 5620 }, 5621 5622 initialize: function (options) { 5623 setOptions(this, options); 5624 5625 this._attributions = {}; 5626 }, 5627 5628 onAdd: function (map) { 5629 map.attributionControl = this; 5630 this._container = create$1('div', 'leaflet-control-attribution'); 5631 disableClickPropagation(this._container); 5632 5633 // TODO ugly, refactor 5634 for (var i in map._layers) { 5635 if (map._layers[i].getAttribution) { 5636 this.addAttribution(map._layers[i].getAttribution()); 5637 } 5638 } 5639 5640 this._update(); 5641 5642 return this._container; 5643 }, 5644 5645 // @method setPrefix(prefix: String): this 5646 // Sets the text before the attributions. 5647 setPrefix: function (prefix) { 5648 this.options.prefix = prefix; 5649 this._update(); 5650 return this; 5651 }, 5652 5653 // @method addAttribution(text: String): this 5654 // Adds an attribution text (e.g. `'Vector data © Mapbox'`). 5655 addAttribution: function (text) { 5656 if (!text) { return this; } 5657 5658 if (!this._attributions[text]) { 5659 this._attributions[text] = 0; 5660 } 5661 this._attributions[text]++; 5662 5663 this._update(); 5664 5665 return this; 5666 }, 5667 5668 // @method removeAttribution(text: String): this 5669 // Removes an attribution text. 5670 removeAttribution: function (text) { 5671 if (!text) { return this; } 5672 5673 if (this._attributions[text]) { 5674 this._attributions[text]--; 5675 this._update(); 5676 } 5677 5678 return this; 5679 }, 5680 5681 _update: function () { 5682 if (!this._map) { return; } 5683 5684 var attribs = []; 5685 5686 for (var i in this._attributions) { 5687 if (this._attributions[i]) { 5688 attribs.push(i); 5689 } 5690 } 5691 5692 var prefixAndAttribs = []; 5693 5694 if (this.options.prefix) { 5695 prefixAndAttribs.push(this.options.prefix); 5696 } 5697 if (attribs.length) { 5698 prefixAndAttribs.push(attribs.join(', ')); 5699 } 5700 5701 this._container.innerHTML = prefixAndAttribs.join(' | '); 5702 } 5703 }); 5704 5705 // @namespace Map 5706 // @section Control options 5707 // @option attributionControl: Boolean = true 5708 // Whether a [attribution control](#control-attribution) is added to the map by default. 5709 Map.mergeOptions({ 5710 attributionControl: true 5711 }); 5712 5713 Map.addInitHook(function () { 5714 if (this.options.attributionControl) { 5715 new Attribution().addTo(this); 5716 } 5717 }); 5718 5719 // @namespace Control.Attribution 5720 // @factory L.control.attribution(options: Control.Attribution options) 5721 // Creates an attribution control. 5722 var attribution = function (options) { 5723 return new Attribution(options); 5724 }; 5725 5726 Control.Layers = Layers; 5727 Control.Zoom = Zoom; 5728 Control.Scale = Scale; 5729 Control.Attribution = Attribution; 5730 5731 control.layers = layers; 5732 control.zoom = zoom; 5733 control.scale = scale; 5734 control.attribution = attribution; 5735 5736 /* 5737 L.Handler is a base class for handler classes that are used internally to inject 5738 interaction features like dragging to classes like Map and Marker. 5739 */ 5740 5741 // @class Handler 5742 // @aka L.Handler 5743 // Abstract class for map interaction handlers 5744 5745 var Handler = Class.extend({ 5746 initialize: function (map) { 5747 this._map = map; 5748 }, 5749 5750 // @method enable(): this 5751 // Enables the handler 5752 enable: function () { 5753 if (this._enabled) { return this; } 5754 5755 this._enabled = true; 5756 this.addHooks(); 5757 return this; 5758 }, 5759 5760 // @method disable(): this 5761 // Disables the handler 5762 disable: function () { 5763 if (!this._enabled) { return this; } 5764 5765 this._enabled = false; 5766 this.removeHooks(); 5767 return this; 5768 }, 5769 5770 // @method enabled(): Boolean 5771 // Returns `true` if the handler is enabled 5772 enabled: function () { 5773 return !!this._enabled; 5774 } 5775 5776 // @section Extension methods 5777 // Classes inheriting from `Handler` must implement the two following methods: 5778 // @method addHooks() 5779 // Called when the handler is enabled, should add event hooks. 5780 // @method removeHooks() 5781 // Called when the handler is disabled, should remove the event hooks added previously. 5782 }); 5783 5784 // @section There is static function which can be called without instantiating L.Handler: 5785 // @function addTo(map: Map, name: String): this 5786 // Adds a new Handler to the given map with the given name. 5787 Handler.addTo = function (map, name) { 5788 map.addHandler(name, this); 5789 return this; 5790 }; 5791 5792 var Mixin = {Events: Events}; 5793 5794 /* 5795 * @class Draggable 5796 * @aka L.Draggable 5797 * @inherits Evented 5798 * 5799 * A class for making DOM elements draggable (including touch support). 5800 * Used internally for map and marker dragging. Only works for elements 5801 * that were positioned with [`L.DomUtil.setPosition`](#domutil-setposition). 5802 * 5803 * @example 5804 * ```js 5805 * var draggable = new L.Draggable(elementToDrag); 5806 * draggable.enable(); 5807 * ``` 5808 */ 5809 5810 var START = touch ? 'touchstart mousedown' : 'mousedown'; 5811 var END = { 5812 mousedown: 'mouseup', 5813 touchstart: 'touchend', 5814 pointerdown: 'touchend', 5815 MSPointerDown: 'touchend' 5816 }; 5817 var MOVE = { 5818 mousedown: 'mousemove', 5819 touchstart: 'touchmove', 5820 pointerdown: 'touchmove', 5821 MSPointerDown: 'touchmove' 5822 }; 5823 5824 5825 var Draggable = Evented.extend({ 5826 5827 options: { 5828 // @section 5829 // @aka Draggable options 5830 // @option clickTolerance: Number = 3 5831 // The max number of pixels a user can shift the mouse pointer during a click 5832 // for it to be considered a valid click (as opposed to a mouse drag). 5833 clickTolerance: 3 5834 }, 5835 5836 // @constructor L.Draggable(el: HTMLElement, dragHandle?: HTMLElement, preventOutline?: Boolean, options?: Draggable options) 5837 // Creates a `Draggable` object for moving `el` when you start dragging the `dragHandle` element (equals `el` itself by default). 5838 initialize: function (element, dragStartTarget, preventOutline$$1, options) { 5839 setOptions(this, options); 5840 5841 this._element = element; 5842 this._dragStartTarget = dragStartTarget || element; 5843 this._preventOutline = preventOutline$$1; 5844 }, 5845 5846 // @method enable() 5847 // Enables the dragging ability 5848 enable: function () { 5849 if (this._enabled) { return; } 5850 5851 on(this._dragStartTarget, START, this._onDown, this); 5852 5853 this._enabled = true; 5854 }, 5855 5856 // @method disable() 5857 // Disables the dragging ability 5858 disable: function () { 5859 if (!this._enabled) { return; } 5860 5861 // If we're currently dragging this draggable, 5862 // disabling it counts as first ending the drag. 5863 if (Draggable._dragging === this) { 5864 this.finishDrag(); 5865 } 5866 5867 off(this._dragStartTarget, START, this._onDown, this); 5868 5869 this._enabled = false; 5870 this._moved = false; 5871 }, 5872 5873 _onDown: function (e) { 5874 // Ignore simulated events, since we handle both touch and 5875 // mouse explicitly; otherwise we risk getting duplicates of 5876 // touch events, see #4315. 5877 // Also ignore the event if disabled; this happens in IE11 5878 // under some circumstances, see #3666. 5879 if (e._simulated || !this._enabled) { return; } 5880 5881 this._moved = false; 5882 5883 if (hasClass(this._element, 'leaflet-zoom-anim')) { return; } 5884 5885 if (Draggable._dragging || e.shiftKey || ((e.which !== 1) && (e.button !== 1) && !e.touches)) { return; } 5886 Draggable._dragging = this; // Prevent dragging multiple objects at once. 5887 5888 if (this._preventOutline) { 5889 preventOutline(this._element); 5890 } 5891 5892 disableImageDrag(); 5893 disableTextSelection(); 5894 5895 if (this._moving) { return; } 5896 5897 // @event down: Event 5898 // Fired when a drag is about to start. 5899 this.fire('down'); 5900 5901 var first = e.touches ? e.touches[0] : e, 5902 sizedParent = getSizedParentNode(this._element); 5903 5904 this._startPoint = new Point(first.clientX, first.clientY); 5905 5906 // Cache the scale, so that we can continuously compensate for it during drag (_onMove). 5907 this._parentScale = getScale(sizedParent); 5908 5909 on(document, MOVE[e.type], this._onMove, this); 5910 on(document, END[e.type], this._onUp, this); 5911 }, 5912 5913 _onMove: function (e) { 5914 // Ignore simulated events, since we handle both touch and 5915 // mouse explicitly; otherwise we risk getting duplicates of 5916 // touch events, see #4315. 5917 // Also ignore the event if disabled; this happens in IE11 5918 // under some circumstances, see #3666. 5919 if (e._simulated || !this._enabled) { return; } 5920 5921 if (e.touches && e.touches.length > 1) { 5922 this._moved = true; 5923 return; 5924 } 5925 5926 var first = (e.touches && e.touches.length === 1 ? e.touches[0] : e), 5927 offset = new Point(first.clientX, first.clientY)._subtract(this._startPoint); 5928 5929 if (!offset.x && !offset.y) { return; } 5930 if (Math.abs(offset.x) + Math.abs(offset.y) < this.options.clickTolerance) { return; } 5931 5932 // We assume that the parent container's position, border and scale do not change for the duration of the drag. 5933 // Therefore there is no need to account for the position and border (they are eliminated by the subtraction) 5934 // and we can use the cached value for the scale. 5935 offset.x /= this._parentScale.x; 5936 offset.y /= this._parentScale.y; 5937 5938 preventDefault(e); 5939 5940 if (!this._moved) { 5941 // @event dragstart: Event 5942 // Fired when a drag starts 5943 this.fire('dragstart'); 5944 5945 this._moved = true; 5946 this._startPos = getPosition(this._element).subtract(offset); 5947 5948 addClass(document.body, 'leaflet-dragging'); 5949 5950 this._lastTarget = e.target || e.srcElement; 5951 // IE and Edge do not give the <use> element, so fetch it 5952 // if necessary 5953 if ((window.SVGElementInstance) && (this._lastTarget instanceof SVGElementInstance)) { 5954 this._lastTarget = this._lastTarget.correspondingUseElement; 5955 } 5956 addClass(this._lastTarget, 'leaflet-drag-target'); 5957 } 5958 5959 this._newPos = this._startPos.add(offset); 5960 this._moving = true; 5961 5962 cancelAnimFrame(this._animRequest); 5963 this._lastEvent = e; 5964 this._animRequest = requestAnimFrame(this._updatePosition, this, true); 5965 }, 5966 5967 _updatePosition: function () { 5968 var e = {originalEvent: this._lastEvent}; 5969 5970 // @event predrag: Event 5971 // Fired continuously during dragging *before* each corresponding 5972 // update of the element's position. 5973 this.fire('predrag', e); 5974 setPosition(this._element, this._newPos); 5975 5976 // @event drag: Event 5977 // Fired continuously during dragging. 5978 this.fire('drag', e); 5979 }, 5980 5981 _onUp: function (e) { 5982 // Ignore simulated events, since we handle both touch and 5983 // mouse explicitly; otherwise we risk getting duplicates of 5984 // touch events, see #4315. 5985 // Also ignore the event if disabled; this happens in IE11 5986 // under some circumstances, see #3666. 5987 if (e._simulated || !this._enabled) { return; } 5988 this.finishDrag(); 5989 }, 5990 5991 finishDrag: function () { 5992 removeClass(document.body, 'leaflet-dragging'); 5993 5994 if (this._lastTarget) { 5995 removeClass(this._lastTarget, 'leaflet-drag-target'); 5996 this._lastTarget = null; 5997 } 5998 5999 for (var i in MOVE) { 6000 off(document, MOVE[i], this._onMove, this); 6001 off(document, END[i], this._onUp, this); 6002 } 6003 6004 enableImageDrag(); 6005 enableTextSelection(); 6006 6007 if (this._moved && this._moving) { 6008 // ensure drag is not fired after dragend 6009 cancelAnimFrame(this._animRequest); 6010 6011 // @event dragend: DragEndEvent 6012 // Fired when the drag ends. 6013 this.fire('dragend', { 6014 distance: this._newPos.distanceTo(this._startPos) 6015 }); 6016 } 6017 6018 this._moving = false; 6019 Draggable._dragging = false; 6020 } 6021 6022 }); 6023 6024 /* 6025 * @namespace LineUtil 6026 * 6027 * Various utility functions for polyline points processing, used by Leaflet internally to make polylines lightning-fast. 6028 */ 6029 6030 // Simplify polyline with vertex reduction and Douglas-Peucker simplification. 6031 // Improves rendering performance dramatically by lessening the number of points to draw. 6032 6033 // @function simplify(points: Point[], tolerance: Number): Point[] 6034 // Dramatically reduces the number of points in a polyline while retaining 6035 // its shape and returns a new array of simplified points, using the 6036 // [Douglas-Peucker algorithm](http://en.wikipedia.org/wiki/Douglas-Peucker_algorithm). 6037 // Used for a huge performance boost when processing/displaying Leaflet polylines for 6038 // each zoom level and also reducing visual noise. tolerance affects the amount of 6039 // simplification (lesser value means higher quality but slower and with more points). 6040 // Also released as a separated micro-library [Simplify.js](http://mourner.github.com/simplify-js/). 6041 function simplify(points, tolerance) { 6042 if (!tolerance || !points.length) { 6043 return points.slice(); 6044 } 6045 6046 var sqTolerance = tolerance * tolerance; 6047 6048 // stage 1: vertex reduction 6049 points = _reducePoints(points, sqTolerance); 6050 6051 // stage 2: Douglas-Peucker simplification 6052 points = _simplifyDP(points, sqTolerance); 6053 6054 return points; 6055 } 6056 6057 // @function pointToSegmentDistance(p: Point, p1: Point, p2: Point): Number 6058 // Returns the distance between point `p` and segment `p1` to `p2`. 6059 function pointToSegmentDistance(p, p1, p2) { 6060 return Math.sqrt(_sqClosestPointOnSegment(p, p1, p2, true)); 6061 } 6062 6063 // @function closestPointOnSegment(p: Point, p1: Point, p2: Point): Number 6064 // Returns the closest point from a point `p` on a segment `p1` to `p2`. 6065 function closestPointOnSegment(p, p1, p2) { 6066 return _sqClosestPointOnSegment(p, p1, p2); 6067 } 6068 6069 // Douglas-Peucker simplification, see http://en.wikipedia.org/wiki/Douglas-Peucker_algorithm 6070 function _simplifyDP(points, sqTolerance) { 6071 6072 var len = points.length, 6073 ArrayConstructor = typeof Uint8Array !== undefined + '' ? Uint8Array : Array, 6074 markers = new ArrayConstructor(len); 6075 6076 markers[0] = markers[len - 1] = 1; 6077 6078 _simplifyDPStep(points, markers, sqTolerance, 0, len - 1); 6079 6080 var i, 6081 newPoints = []; 6082 6083 for (i = 0; i < len; i++) { 6084 if (markers[i]) { 6085 newPoints.push(points[i]); 6086 } 6087 } 6088 6089 return newPoints; 6090 } 6091 6092 function _simplifyDPStep(points, markers, sqTolerance, first, last) { 6093 6094 var maxSqDist = 0, 6095 index, i, sqDist; 6096 6097 for (i = first + 1; i <= last - 1; i++) { 6098 sqDist = _sqClosestPointOnSegment(points[i], points[first], points[last], true); 6099 6100 if (sqDist > maxSqDist) { 6101 index = i; 6102 maxSqDist = sqDist; 6103 } 6104 } 6105 6106 if (maxSqDist > sqTolerance) { 6107 markers[index] = 1; 6108 6109 _simplifyDPStep(points, markers, sqTolerance, first, index); 6110 _simplifyDPStep(points, markers, sqTolerance, index, last); 6111 } 6112 } 6113 6114 // reduce points that are too close to each other to a single point 6115 function _reducePoints(points, sqTolerance) { 6116 var reducedPoints = [points[0]]; 6117 6118 for (var i = 1, prev = 0, len = points.length; i < len; i++) { 6119 if (_sqDist(points[i], points[prev]) > sqTolerance) { 6120 reducedPoints.push(points[i]); 6121 prev = i; 6122 } 6123 } 6124 if (prev < len - 1) { 6125 reducedPoints.push(points[len - 1]); 6126 } 6127 return reducedPoints; 6128 } 6129 6130 var _lastCode; 6131 6132 // @function clipSegment(a: Point, b: Point, bounds: Bounds, useLastCode?: Boolean, round?: Boolean): Point[]|Boolean 6133 // Clips the segment a to b by rectangular bounds with the 6134 // [Cohen-Sutherland algorithm](https://en.wikipedia.org/wiki/Cohen%E2%80%93Sutherland_algorithm) 6135 // (modifying the segment points directly!). Used by Leaflet to only show polyline 6136 // points that are on the screen or near, increasing performance. 6137 function clipSegment(a, b, bounds, useLastCode, round) { 6138 var codeA = useLastCode ? _lastCode : _getBitCode(a, bounds), 6139 codeB = _getBitCode(b, bounds), 6140 6141 codeOut, p, newCode; 6142 6143 // save 2nd code to avoid calculating it on the next segment 6144 _lastCode = codeB; 6145 6146 while (true) { 6147 // if a,b is inside the clip window (trivial accept) 6148 if (!(codeA | codeB)) { 6149 return [a, b]; 6150 } 6151 6152 // if a,b is outside the clip window (trivial reject) 6153 if (codeA & codeB) { 6154 return false; 6155 } 6156 6157 // other cases 6158 codeOut = codeA || codeB; 6159 p = _getEdgeIntersection(a, b, codeOut, bounds, round); 6160 newCode = _getBitCode(p, bounds); 6161 6162 if (codeOut === codeA) { 6163 a = p; 6164 codeA = newCode; 6165 } else { 6166 b = p; 6167 codeB = newCode; 6168 } 6169 } 6170 } 6171 6172 function _getEdgeIntersection(a, b, code, bounds, round) { 6173 var dx = b.x - a.x, 6174 dy = b.y - a.y, 6175 min = bounds.min, 6176 max = bounds.max, 6177 x, y; 6178 6179 if (code & 8) { // top 6180 x = a.x + dx * (max.y - a.y) / dy; 6181 y = max.y; 6182 6183 } else if (code & 4) { // bottom 6184 x = a.x + dx * (min.y - a.y) / dy; 6185 y = min.y; 6186 6187 } else if (code & 2) { // right 6188 x = max.x; 6189 y = a.y + dy * (max.x - a.x) / dx; 6190 6191 } else if (code & 1) { // left 6192 x = min.x; 6193 y = a.y + dy * (min.x - a.x) / dx; 6194 } 6195 6196 return new Point(x, y, round); 6197 } 6198 6199 function _getBitCode(p, bounds) { 6200 var code = 0; 6201 6202 if (p.x < bounds.min.x) { // left 6203 code |= 1; 6204 } else if (p.x > bounds.max.x) { // right 6205 code |= 2; 6206 } 6207 6208 if (p.y < bounds.min.y) { // bottom 6209 code |= 4; 6210 } else if (p.y > bounds.max.y) { // top 6211 code |= 8; 6212 } 6213 6214 return code; 6215 } 6216 6217 // square distance (to avoid unnecessary Math.sqrt calls) 6218 function _sqDist(p1, p2) { 6219 var dx = p2.x - p1.x, 6220 dy = p2.y - p1.y; 6221 return dx * dx + dy * dy; 6222 } 6223 6224 // return closest point on segment or distance to that point 6225 function _sqClosestPointOnSegment(p, p1, p2, sqDist) { 6226 var x = p1.x, 6227 y = p1.y, 6228 dx = p2.x - x, 6229 dy = p2.y - y, 6230 dot = dx * dx + dy * dy, 6231 t; 6232 6233 if (dot > 0) { 6234 t = ((p.x - x) * dx + (p.y - y) * dy) / dot; 6235 6236 if (t > 1) { 6237 x = p2.x; 6238 y = p2.y; 6239 } else if (t > 0) { 6240 x += dx * t; 6241 y += dy * t; 6242 } 6243 } 6244 6245 dx = p.x - x; 6246 dy = p.y - y; 6247 6248 return sqDist ? dx * dx + dy * dy : new Point(x, y); 6249 } 6250 6251 6252 // @function isFlat(latlngs: LatLng[]): Boolean 6253 // Returns true if `latlngs` is a flat array, false is nested. 6254 function isFlat(latlngs) { 6255 return !isArray(latlngs[0]) || (typeof latlngs[0][0] !== 'object' && typeof latlngs[0][0] !== 'undefined'); 6256 } 6257 6258 function _flat(latlngs) { 6259 console.warn('Deprecated use of _flat, please use L.LineUtil.isFlat instead.'); 6260 return isFlat(latlngs); 6261 } 6262 6263 6264 var LineUtil = (Object.freeze || Object)({ 6265 simplify: simplify, 6266 pointToSegmentDistance: pointToSegmentDistance, 6267 closestPointOnSegment: closestPointOnSegment, 6268 clipSegment: clipSegment, 6269 _getEdgeIntersection: _getEdgeIntersection, 6270 _getBitCode: _getBitCode, 6271 _sqClosestPointOnSegment: _sqClosestPointOnSegment, 6272 isFlat: isFlat, 6273 _flat: _flat 6274 }); 6275 6276 /* 6277 * @namespace PolyUtil 6278 * Various utility functions for polygon geometries. 6279 */ 6280 6281 /* @function clipPolygon(points: Point[], bounds: Bounds, round?: Boolean): Point[] 6282 * Clips the polygon geometry defined by the given `points` by the given bounds (using the [Sutherland-Hodgman algorithm](https://en.wikipedia.org/wiki/Sutherland%E2%80%93Hodgman_algorithm)). 6283 * Used by Leaflet to only show polygon points that are on the screen or near, increasing 6284 * performance. Note that polygon points needs different algorithm for clipping 6285 * than polyline, so there's a separate method for it. 6286 */ 6287 function clipPolygon(points, bounds, round) { 6288 var clippedPoints, 6289 edges = [1, 4, 2, 8], 6290 i, j, k, 6291 a, b, 6292 len, edge, p; 6293 6294 for (i = 0, len = points.length; i < len; i++) { 6295 points[i]._code = _getBitCode(points[i], bounds); 6296 } 6297 6298 // for each edge (left, bottom, right, top) 6299 for (k = 0; k < 4; k++) { 6300 edge = edges[k]; 6301 clippedPoints = []; 6302 6303 for (i = 0, len = points.length, j = len - 1; i < len; j = i++) { 6304 a = points[i]; 6305 b = points[j]; 6306 6307 // if a is inside the clip window 6308 if (!(a._code & edge)) { 6309 // if b is outside the clip window (a->b goes out of screen) 6310 if (b._code & edge) { 6311 p = _getEdgeIntersection(b, a, edge, bounds, round); 6312 p._code = _getBitCode(p, bounds); 6313 clippedPoints.push(p); 6314 } 6315 clippedPoints.push(a); 6316 6317 // else if b is inside the clip window (a->b enters the screen) 6318 } else if (!(b._code & edge)) { 6319 p = _getEdgeIntersection(b, a, edge, bounds, round); 6320 p._code = _getBitCode(p, bounds); 6321 clippedPoints.push(p); 6322 } 6323 } 6324 points = clippedPoints; 6325 } 6326 6327 return points; 6328 } 6329 6330 6331 var PolyUtil = (Object.freeze || Object)({ 6332 clipPolygon: clipPolygon 6333 }); 6334 6335 /* 6336 * @namespace Projection 6337 * @section 6338 * Leaflet comes with a set of already defined Projections out of the box: 6339 * 6340 * @projection L.Projection.LonLat 6341 * 6342 * Equirectangular, or Plate Carree projection — the most simple projection, 6343 * mostly used by GIS enthusiasts. Directly maps `x` as longitude, and `y` as 6344 * latitude. Also suitable for flat worlds, e.g. game maps. Used by the 6345 * `EPSG:4326` and `Simple` CRS. 6346 */ 6347 6348 var LonLat = { 6349 project: function (latlng) { 6350 return new Point(latlng.lng, latlng.lat); 6351 }, 6352 6353 unproject: function (point) { 6354 return new LatLng(point.y, point.x); 6355 }, 6356 6357 bounds: new Bounds([-180, -90], [180, 90]) 6358 }; 6359 6360 /* 6361 * @namespace Projection 6362 * @projection L.Projection.Mercator 6363 * 6364 * Elliptical Mercator projection — more complex than Spherical Mercator. Assumes that Earth is an ellipsoid. Used by the EPSG:3395 CRS. 6365 */ 6366 6367 var Mercator = { 6368 R: 6378137, 6369 R_MINOR: 6356752.314245179, 6370 6371 bounds: new Bounds([-20037508.34279, -15496570.73972], [20037508.34279, 18764656.23138]), 6372 6373 project: function (latlng) { 6374 var d = Math.PI / 180, 6375 r = this.R, 6376 y = latlng.lat * d, 6377 tmp = this.R_MINOR / r, 6378 e = Math.sqrt(1 - tmp * tmp), 6379 con = e * Math.sin(y); 6380 6381 var ts = Math.tan(Math.PI / 4 - y / 2) / Math.pow((1 - con) / (1 + con), e / 2); 6382 y = -r * Math.log(Math.max(ts, 1E-10)); 6383 6384 return new Point(latlng.lng * d * r, y); 6385 }, 6386 6387 unproject: function (point) { 6388 var d = 180 / Math.PI, 6389 r = this.R, 6390 tmp = this.R_MINOR / r, 6391 e = Math.sqrt(1 - tmp * tmp), 6392 ts = Math.exp(-point.y / r), 6393 phi = Math.PI / 2 - 2 * Math.atan(ts); 6394 6395 for (var i = 0, dphi = 0.1, con; i < 15 && Math.abs(dphi) > 1e-7; i++) { 6396 con = e * Math.sin(phi); 6397 con = Math.pow((1 - con) / (1 + con), e / 2); 6398 dphi = Math.PI / 2 - 2 * Math.atan(ts * con) - phi; 6399 phi += dphi; 6400 } 6401 6402 return new LatLng(phi * d, point.x * d / r); 6403 } 6404 }; 6405 6406 /* 6407 * @class Projection 6408 6409 * An object with methods for projecting geographical coordinates of the world onto 6410 * a flat surface (and back). See [Map projection](http://en.wikipedia.org/wiki/Map_projection). 6411 6412 * @property bounds: Bounds 6413 * The bounds (specified in CRS units) where the projection is valid 6414 6415 * @method project(latlng: LatLng): Point 6416 * Projects geographical coordinates into a 2D point. 6417 * Only accepts actual `L.LatLng` instances, not arrays. 6418 6419 * @method unproject(point: Point): LatLng 6420 * The inverse of `project`. Projects a 2D point into a geographical location. 6421 * Only accepts actual `L.Point` instances, not arrays. 6422 6423 * Note that the projection instances do not inherit from Leafet's `Class` object, 6424 * and can't be instantiated. Also, new classes can't inherit from them, 6425 * and methods can't be added to them with the `include` function. 6426 6427 */ 6428 6429 6430 6431 6432 var index = (Object.freeze || Object)({ 6433 LonLat: LonLat, 6434 Mercator: Mercator, 6435 SphericalMercator: SphericalMercator 6436 }); 6437 6438 /* 6439 * @namespace CRS 6440 * @crs L.CRS.EPSG3395 6441 * 6442 * Rarely used by some commercial tile providers. Uses Elliptical Mercator projection. 6443 */ 6444 var EPSG3395 = extend({}, Earth, { 6445 code: 'EPSG:3395', 6446 projection: Mercator, 6447 6448 transformation: (function () { 6449 var scale = 0.5 / (Math.PI * Mercator.R); 6450 return toTransformation(scale, 0.5, -scale, 0.5); 6451 }()) 6452 }); 6453 6454 /* 6455 * @namespace CRS 6456 * @crs L.CRS.EPSG4326 6457 * 6458 * A common CRS among GIS enthusiasts. Uses simple Equirectangular projection. 6459 * 6460 * Leaflet 1.0.x complies with the [TMS coordinate scheme for EPSG:4326](https://wiki.osgeo.org/wiki/Tile_Map_Service_Specification#global-geodetic), 6461 * which is a breaking change from 0.7.x behaviour. If you are using a `TileLayer` 6462 * with this CRS, ensure that there are two 256x256 pixel tiles covering the 6463 * whole earth at zoom level zero, and that the tile coordinate origin is (-180,+90), 6464 * or (-180,-90) for `TileLayer`s with [the `tms` option](#tilelayer-tms) set. 6465 */ 6466 6467 var EPSG4326 = extend({}, Earth, { 6468 code: 'EPSG:4326', 6469 projection: LonLat, 6470 transformation: toTransformation(1 / 180, 1, -1 / 180, 0.5) 6471 }); 6472 6473 /* 6474 * @namespace CRS 6475 * @crs L.CRS.Simple 6476 * 6477 * A simple CRS that maps longitude and latitude into `x` and `y` directly. 6478 * May be used for maps of flat surfaces (e.g. game maps). Note that the `y` 6479 * axis should still be inverted (going from bottom to top). `distance()` returns 6480 * simple euclidean distance. 6481 */ 6482 6483 var Simple = extend({}, CRS, { 6484 projection: LonLat, 6485 transformation: toTransformation(1, 0, -1, 0), 6486 6487 scale: function (zoom) { 6488 return Math.pow(2, zoom); 6489 }, 6490 6491 zoom: function (scale) { 6492 return Math.log(scale) / Math.LN2; 6493 }, 6494 6495 distance: function (latlng1, latlng2) { 6496 var dx = latlng2.lng - latlng1.lng, 6497 dy = latlng2.lat - latlng1.lat; 6498 6499 return Math.sqrt(dx * dx + dy * dy); 6500 }, 6501 6502 infinite: true 6503 }); 6504 6505 CRS.Earth = Earth; 6506 CRS.EPSG3395 = EPSG3395; 6507 CRS.EPSG3857 = EPSG3857; 6508 CRS.EPSG900913 = EPSG900913; 6509 CRS.EPSG4326 = EPSG4326; 6510 CRS.Simple = Simple; 6511 6512 /* 6513 * @class Layer 6514 * @inherits Evented 6515 * @aka L.Layer 6516 * @aka ILayer 6517 * 6518 * A set of methods from the Layer base class that all Leaflet layers use. 6519 * Inherits all methods, options and events from `L.Evented`. 6520 * 6521 * @example 6522 * 6523 * ```js 6524 * var layer = L.marker(latlng).addTo(map); 6525 * layer.addTo(map); 6526 * layer.remove(); 6527 * ``` 6528 * 6529 * @event add: Event 6530 * Fired after the layer is added to a map 6531 * 6532 * @event remove: Event 6533 * Fired after the layer is removed from a map 6534 */ 6535 6536 6537 var Layer = Evented.extend({ 6538 6539 // Classes extending `L.Layer` will inherit the following options: 6540 options: { 6541 // @option pane: String = 'overlayPane' 6542 // By default the layer will be added to the map's [overlay pane](#map-overlaypane). Overriding this option will cause the layer to be placed on another pane by default. 6543 pane: 'overlayPane', 6544 6545 // @option attribution: String = null 6546 // String to be shown in the attribution control, e.g. "© OpenStreetMap contributors". It describes the layer data and is often a legal obligation towards copyright holders and tile providers. 6547 attribution: null, 6548 6549 bubblingMouseEvents: true 6550 }, 6551 6552 /* @section 6553 * Classes extending `L.Layer` will inherit the following methods: 6554 * 6555 * @method addTo(map: Map|LayerGroup): this 6556 * Adds the layer to the given map or layer group. 6557 */ 6558 addTo: function (map) { 6559 map.addLayer(this); 6560 return this; 6561 }, 6562 6563 // @method remove: this 6564 // Removes the layer from the map it is currently active on. 6565 remove: function () { 6566 return this.removeFrom(this._map || this._mapToAdd); 6567 }, 6568 6569 // @method removeFrom(map: Map): this 6570 // Removes the layer from the given map 6571 removeFrom: function (obj) { 6572 if (obj) { 6573 obj.removeLayer(this); 6574 } 6575 return this; 6576 }, 6577 6578 // @method getPane(name? : String): HTMLElement 6579 // Returns the `HTMLElement` representing the named pane on the map. If `name` is omitted, returns the pane for this layer. 6580 getPane: function (name) { 6581 return this._map.getPane(name ? (this.options[name] || name) : this.options.pane); 6582 }, 6583 6584 addInteractiveTarget: function (targetEl) { 6585 this._map._targets[stamp(targetEl)] = this; 6586 return this; 6587 }, 6588 6589 removeInteractiveTarget: function (targetEl) { 6590 delete this._map._targets[stamp(targetEl)]; 6591 return this; 6592 }, 6593 6594 // @method getAttribution: String 6595 // Used by the `attribution control`, returns the [attribution option](#gridlayer-attribution). 6596 getAttribution: function () { 6597 return this.options.attribution; 6598 }, 6599 6600 _layerAdd: function (e) { 6601 var map = e.target; 6602 6603 // check in case layer gets added and then removed before the map is ready 6604 if (!map.hasLayer(this)) { return; } 6605 6606 this._map = map; 6607 this._zoomAnimated = map._zoomAnimated; 6608 6609 if (this.getEvents) { 6610 var events = this.getEvents(); 6611 map.on(events, this); 6612 this.once('remove', function () { 6613 map.off(events, this); 6614 }, this); 6615 } 6616 6617 this.onAdd(map); 6618 6619 if (this.getAttribution && map.attributionControl) { 6620 map.attributionControl.addAttribution(this.getAttribution()); 6621 } 6622 6623 this.fire('add'); 6624 map.fire('layeradd', {layer: this}); 6625 } 6626 }); 6627 6628 /* @section Extension methods 6629 * @uninheritable 6630 * 6631 * Every layer should extend from `L.Layer` and (re-)implement the following methods. 6632 * 6633 * @method onAdd(map: Map): this 6634 * Should contain code that creates DOM elements for the layer, adds them to `map panes` where they should belong and puts listeners on relevant map events. Called on [`map.addLayer(layer)`](#map-addlayer). 6635 * 6636 * @method onRemove(map: Map): this 6637 * Should contain all clean up code that removes the layer's elements from the DOM and removes listeners previously added in [`onAdd`](#layer-onadd). Called on [`map.removeLayer(layer)`](#map-removelayer). 6638 * 6639 * @method getEvents(): Object 6640 * This optional method should return an object like `{ viewreset: this._reset }` for [`addEventListener`](#evented-addeventlistener). The event handlers in this object will be automatically added and removed from the map with your layer. 6641 * 6642 * @method getAttribution(): String 6643 * This optional method should return a string containing HTML to be shown on the `Attribution control` whenever the layer is visible. 6644 * 6645 * @method beforeAdd(map: Map): this 6646 * Optional method. Called on [`map.addLayer(layer)`](#map-addlayer), before the layer is added to the map, before events are initialized, without waiting until the map is in a usable state. Use for early initialization only. 6647 */ 6648 6649 6650 /* @namespace Map 6651 * @section Layer events 6652 * 6653 * @event layeradd: LayerEvent 6654 * Fired when a new layer is added to the map. 6655 * 6656 * @event layerremove: LayerEvent 6657 * Fired when some layer is removed from the map 6658 * 6659 * @section Methods for Layers and Controls 6660 */ 6661 Map.include({ 6662 // @method addLayer(layer: Layer): this 6663 // Adds the given layer to the map 6664 addLayer: function (layer) { 6665 if (!layer._layerAdd) { 6666 throw new Error('The provided object is not a Layer.'); 6667 } 6668 6669 var id = stamp(layer); 6670 if (this._layers[id]) { return this; } 6671 this._layers[id] = layer; 6672 6673 layer._mapToAdd = this; 6674 6675 if (layer.beforeAdd) { 6676 layer.beforeAdd(this); 6677 } 6678 6679 this.whenReady(layer._layerAdd, layer); 6680 6681 return this; 6682 }, 6683 6684 // @method removeLayer(layer: Layer): this 6685 // Removes the given layer from the map. 6686 removeLayer: function (layer) { 6687 var id = stamp(layer); 6688 6689 if (!this._layers[id]) { return this; } 6690 6691 if (this._loaded) { 6692 layer.onRemove(this); 6693 } 6694 6695 if (layer.getAttribution && this.attributionControl) { 6696 this.attributionControl.removeAttribution(layer.getAttribution()); 6697 } 6698 6699 delete this._layers[id]; 6700 6701 if (this._loaded) { 6702 this.fire('layerremove', {layer: layer}); 6703 layer.fire('remove'); 6704 } 6705 6706 layer._map = layer._mapToAdd = null; 6707 6708 return this; 6709 }, 6710 6711 // @method hasLayer(layer: Layer): Boolean 6712 // Returns `true` if the given layer is currently added to the map 6713 hasLayer: function (layer) { 6714 return !!layer && (stamp(layer) in this._layers); 6715 }, 6716 6717 /* @method eachLayer(fn: Function, context?: Object): this 6718 * Iterates over the layers of the map, optionally specifying context of the iterator function. 6719 * ``` 6720 * map.eachLayer(function(layer){ 6721 * layer.bindPopup('Hello'); 6722 * }); 6723 * ``` 6724 */ 6725 eachLayer: function (method, context) { 6726 for (var i in this._layers) { 6727 method.call(context, this._layers[i]); 6728 } 6729 return this; 6730 }, 6731 6732 _addLayers: function (layers) { 6733 layers = layers ? (isArray(layers) ? layers : [layers]) : []; 6734 6735 for (var i = 0, len = layers.length; i < len; i++) { 6736 this.addLayer(layers[i]); 6737 } 6738 }, 6739 6740 _addZoomLimit: function (layer) { 6741 if (isNaN(layer.options.maxZoom) || !isNaN(layer.options.minZoom)) { 6742 this._zoomBoundLayers[stamp(layer)] = layer; 6743 this._updateZoomLevels(); 6744 } 6745 }, 6746 6747 _removeZoomLimit: function (layer) { 6748 var id = stamp(layer); 6749 6750 if (this._zoomBoundLayers[id]) { 6751 delete this._zoomBoundLayers[id]; 6752 this._updateZoomLevels(); 6753 } 6754 }, 6755 6756 _updateZoomLevels: function () { 6757 var minZoom = Infinity, 6758 maxZoom = -Infinity, 6759 oldZoomSpan = this._getZoomSpan(); 6760 6761 for (var i in this._zoomBoundLayers) { 6762 var options = this._zoomBoundLayers[i].options; 6763 6764 minZoom = options.minZoom === undefined ? minZoom : Math.min(minZoom, options.minZoom); 6765 maxZoom = options.maxZoom === undefined ? maxZoom : Math.max(maxZoom, options.maxZoom); 6766 } 6767 6768 this._layersMaxZoom = maxZoom === -Infinity ? undefined : maxZoom; 6769 this._layersMinZoom = minZoom === Infinity ? undefined : minZoom; 6770 6771 // @section Map state change events 6772 // @event zoomlevelschange: Event 6773 // Fired when the number of zoomlevels on the map is changed due 6774 // to adding or removing a layer. 6775 if (oldZoomSpan !== this._getZoomSpan()) { 6776 this.fire('zoomlevelschange'); 6777 } 6778 6779 if (this.options.maxZoom === undefined && this._layersMaxZoom && this.getZoom() > this._layersMaxZoom) { 6780 this.setZoom(this._layersMaxZoom); 6781 } 6782 if (this.options.minZoom === undefined && this._layersMinZoom && this.getZoom() < this._layersMinZoom) { 6783 this.setZoom(this._layersMinZoom); 6784 } 6785 } 6786 }); 6787 6788 /* 6789 * @class LayerGroup 6790 * @aka L.LayerGroup 6791 * @inherits Layer 6792 * 6793 * Used to group several layers and handle them as one. If you add it to the map, 6794 * any layers added or removed from the group will be added/removed on the map as 6795 * well. Extends `Layer`. 6796 * 6797 * @example 6798 * 6799 * ```js 6800 * L.layerGroup([marker1, marker2]) 6801 * .addLayer(polyline) 6802 * .addTo(map); 6803 * ``` 6804 */ 6805 6806 var LayerGroup = Layer.extend({ 6807 6808 initialize: function (layers, options) { 6809 setOptions(this, options); 6810 6811 this._layers = {}; 6812 6813 var i, len; 6814 6815 if (layers) { 6816 for (i = 0, len = layers.length; i < len; i++) { 6817 this.addLayer(layers[i]); 6818 } 6819 } 6820 }, 6821 6822 // @method addLayer(layer: Layer): this 6823 // Adds the given layer to the group. 6824 addLayer: function (layer) { 6825 var id = this.getLayerId(layer); 6826 6827 this._layers[id] = layer; 6828 6829 if (this._map) { 6830 this._map.addLayer(layer); 6831 } 6832 6833 return this; 6834 }, 6835 6836 // @method removeLayer(layer: Layer): this 6837 // Removes the given layer from the group. 6838 // @alternative 6839 // @method removeLayer(id: Number): this 6840 // Removes the layer with the given internal ID from the group. 6841 removeLayer: function (layer) { 6842 var id = layer in this._layers ? layer : this.getLayerId(layer); 6843 6844 if (this._map && this._layers[id]) { 6845 this._map.removeLayer(this._layers[id]); 6846 } 6847 6848 delete this._layers[id]; 6849 6850 return this; 6851 }, 6852 6853 // @method hasLayer(layer: Layer): Boolean 6854 // Returns `true` if the given layer is currently added to the group. 6855 // @alternative 6856 // @method hasLayer(id: Number): Boolean 6857 // Returns `true` if the given internal ID is currently added to the group. 6858 hasLayer: function (layer) { 6859 return !!layer && (layer in this._layers || this.getLayerId(layer) in this._layers); 6860 }, 6861 6862 // @method clearLayers(): this 6863 // Removes all the layers from the group. 6864 clearLayers: function () { 6865 return this.eachLayer(this.removeLayer, this); 6866 }, 6867 6868 // @method invoke(methodName: String, …): this 6869 // Calls `methodName` on every layer contained in this group, passing any 6870 // additional parameters. Has no effect if the layers contained do not 6871 // implement `methodName`. 6872 invoke: function (methodName) { 6873 var args = Array.prototype.slice.call(arguments, 1), 6874 i, layer; 6875 6876 for (i in this._layers) { 6877 layer = this._layers[i]; 6878 6879 if (layer[methodName]) { 6880 layer[methodName].apply(layer, args); 6881 } 6882 } 6883 6884 return this; 6885 }, 6886 6887 onAdd: function (map) { 6888 this.eachLayer(map.addLayer, map); 6889 }, 6890 6891 onRemove: function (map) { 6892 this.eachLayer(map.removeLayer, map); 6893 }, 6894 6895 // @method eachLayer(fn: Function, context?: Object): this 6896 // Iterates over the layers of the group, optionally specifying context of the iterator function. 6897 // ```js 6898 // group.eachLayer(function (layer) { 6899 // layer.bindPopup('Hello'); 6900 // }); 6901 // ``` 6902 eachLayer: function (method, context) { 6903 for (var i in this._layers) { 6904 method.call(context, this._layers[i]); 6905 } 6906 return this; 6907 }, 6908 6909 // @method getLayer(id: Number): Layer 6910 // Returns the layer with the given internal ID. 6911 getLayer: function (id) { 6912 return this._layers[id]; 6913 }, 6914 6915 // @method getLayers(): Layer[] 6916 // Returns an array of all the layers added to the group. 6917 getLayers: function () { 6918 var layers = []; 6919 this.eachLayer(layers.push, layers); 6920 return layers; 6921 }, 6922 6923 // @method setZIndex(zIndex: Number): this 6924 // Calls `setZIndex` on every layer contained in this group, passing the z-index. 6925 setZIndex: function (zIndex) { 6926 return this.invoke('setZIndex', zIndex); 6927 }, 6928 6929 // @method getLayerId(layer: Layer): Number 6930 // Returns the internal ID for a layer 6931 getLayerId: function (layer) { 6932 return stamp(layer); 6933 } 6934 }); 6935 6936 6937 // @factory L.layerGroup(layers?: Layer[], options?: Object) 6938 // Create a layer group, optionally given an initial set of layers and an `options` object. 6939 var layerGroup = function (layers, options) { 6940 return new LayerGroup(layers, options); 6941 }; 6942 6943 /* 6944 * @class FeatureGroup 6945 * @aka L.FeatureGroup 6946 * @inherits LayerGroup 6947 * 6948 * Extended `LayerGroup` that makes it easier to do the same thing to all its member layers: 6949 * * [`bindPopup`](#layer-bindpopup) binds a popup to all of the layers at once (likewise with [`bindTooltip`](#layer-bindtooltip)) 6950 * * Events are propagated to the `FeatureGroup`, so if the group has an event 6951 * handler, it will handle events from any of the layers. This includes mouse events 6952 * and custom events. 6953 * * Has `layeradd` and `layerremove` events 6954 * 6955 * @example 6956 * 6957 * ```js 6958 * L.featureGroup([marker1, marker2, polyline]) 6959 * .bindPopup('Hello world!') 6960 * .on('click', function() { alert('Clicked on a member of the group!'); }) 6961 * .addTo(map); 6962 * ``` 6963 */ 6964 6965 var FeatureGroup = LayerGroup.extend({ 6966 6967 addLayer: function (layer) { 6968 if (this.hasLayer(layer)) { 6969 return this; 6970 } 6971 6972 layer.addEventParent(this); 6973 6974 LayerGroup.prototype.addLayer.call(this, layer); 6975 6976 // @event layeradd: LayerEvent 6977 // Fired when a layer is added to this `FeatureGroup` 6978 return this.fire('layeradd', {layer: layer}); 6979 }, 6980 6981 removeLayer: function (layer) { 6982 if (!this.hasLayer(layer)) { 6983 return this; 6984 } 6985 if (layer in this._layers) { 6986 layer = this._layers[layer]; 6987 } 6988 6989 layer.removeEventParent(this); 6990 6991 LayerGroup.prototype.removeLayer.call(this, layer); 6992 6993 // @event layerremove: LayerEvent 6994 // Fired when a layer is removed from this `FeatureGroup` 6995 return this.fire('layerremove', {layer: layer}); 6996 }, 6997 6998 // @method setStyle(style: Path options): this 6999 // Sets the given path options to each layer of the group that has a `setStyle` method. 7000 setStyle: function (style) { 7001 return this.invoke('setStyle', style); 7002 }, 7003 7004 // @method bringToFront(): this 7005 // Brings the layer group to the top of all other layers 7006 bringToFront: function () { 7007 return this.invoke('bringToFront'); 7008 }, 7009 7010 // @method bringToBack(): this 7011 // Brings the layer group to the back of all other layers 7012 bringToBack: function () { 7013 return this.invoke('bringToBack'); 7014 }, 7015 7016 // @method getBounds(): LatLngBounds 7017 // Returns the LatLngBounds of the Feature Group (created from bounds and coordinates of its children). 7018 getBounds: function () { 7019 var bounds = new LatLngBounds(); 7020 7021 for (var id in this._layers) { 7022 var layer = this._layers[id]; 7023 bounds.extend(layer.getBounds ? layer.getBounds() : layer.getLatLng()); 7024 } 7025 return bounds; 7026 } 7027 }); 7028 7029 // @factory L.featureGroup(layers: Layer[]) 7030 // Create a feature group, optionally given an initial set of layers. 7031 var featureGroup = function (layers) { 7032 return new FeatureGroup(layers); 7033 }; 7034 7035 /* 7036 * @class Icon 7037 * @aka L.Icon 7038 * 7039 * Represents an icon to provide when creating a marker. 7040 * 7041 * @example 7042 * 7043 * ```js 7044 * var myIcon = L.icon({ 7045 * iconUrl: 'my-icon.png', 7046 * iconRetinaUrl: 'my-icon@2x.png', 7047 * iconSize: [38, 95], 7048 * iconAnchor: [22, 94], 7049 * popupAnchor: [-3, -76], 7050 * shadowUrl: 'my-icon-shadow.png', 7051 * shadowRetinaUrl: 'my-icon-shadow@2x.png', 7052 * shadowSize: [68, 95], 7053 * shadowAnchor: [22, 94] 7054 * }); 7055 * 7056 * L.marker([50.505, 30.57], {icon: myIcon}).addTo(map); 7057 * ``` 7058 * 7059 * `L.Icon.Default` extends `L.Icon` and is the blue icon Leaflet uses for markers by default. 7060 * 7061 */ 7062 7063 var Icon = Class.extend({ 7064 7065 /* @section 7066 * @aka Icon options 7067 * 7068 * @option iconUrl: String = null 7069 * **(required)** The URL to the icon image (absolute or relative to your script path). 7070 * 7071 * @option iconRetinaUrl: String = null 7072 * The URL to a retina sized version of the icon image (absolute or relative to your 7073 * script path). Used for Retina screen devices. 7074 * 7075 * @option iconSize: Point = null 7076 * Size of the icon image in pixels. 7077 * 7078 * @option iconAnchor: Point = null 7079 * The coordinates of the "tip" of the icon (relative to its top left corner). The icon 7080 * will be aligned so that this point is at the marker's geographical location. Centered 7081 * by default if size is specified, also can be set in CSS with negative margins. 7082 * 7083 * @option popupAnchor: Point = [0, 0] 7084 * The coordinates of the point from which popups will "open", relative to the icon anchor. 7085 * 7086 * @option tooltipAnchor: Point = [0, 0] 7087 * The coordinates of the point from which tooltips will "open", relative to the icon anchor. 7088 * 7089 * @option shadowUrl: String = null 7090 * The URL to the icon shadow image. If not specified, no shadow image will be created. 7091 * 7092 * @option shadowRetinaUrl: String = null 7093 * 7094 * @option shadowSize: Point = null 7095 * Size of the shadow image in pixels. 7096 * 7097 * @option shadowAnchor: Point = null 7098 * The coordinates of the "tip" of the shadow (relative to its top left corner) (the same 7099 * as iconAnchor if not specified). 7100 * 7101 * @option className: String = '' 7102 * A custom class name to assign to both icon and shadow images. Empty by default. 7103 */ 7104 7105 options: { 7106 popupAnchor: [0, 0], 7107 tooltipAnchor: [0, 0] 7108 }, 7109 7110 initialize: function (options) { 7111 setOptions(this, options); 7112 }, 7113 7114 // @method createIcon(oldIcon?: HTMLElement): HTMLElement 7115 // Called internally when the icon has to be shown, returns a `<img>` HTML element 7116 // styled according to the options. 7117 createIcon: function (oldIcon) { 7118 return this._createIcon('icon', oldIcon); 7119 }, 7120 7121 // @method createShadow(oldIcon?: HTMLElement): HTMLElement 7122 // As `createIcon`, but for the shadow beneath it. 7123 createShadow: function (oldIcon) { 7124 return this._createIcon('shadow', oldIcon); 7125 }, 7126 7127 _createIcon: function (name, oldIcon) { 7128 var src = this._getIconUrl(name); 7129 7130 if (!src) { 7131 if (name === 'icon') { 7132 throw new Error('iconUrl not set in Icon options (see the docs).'); 7133 } 7134 return null; 7135 } 7136 7137 var img = this._createImg(src, oldIcon && oldIcon.tagName === 'IMG' ? oldIcon : null); 7138 this._setIconStyles(img, name); 7139 7140 return img; 7141 }, 7142 7143 _setIconStyles: function (img, name) { 7144 var options = this.options; 7145 var sizeOption = options[name + 'Size']; 7146 7147 if (typeof sizeOption === 'number') { 7148 sizeOption = [sizeOption, sizeOption]; 7149 } 7150 7151 var size = toPoint(sizeOption), 7152 anchor = toPoint(name === 'shadow' && options.shadowAnchor || options.iconAnchor || 7153 size && size.divideBy(2, true)); 7154 7155 img.className = 'leaflet-marker-' + name + ' ' + (options.className || ''); 7156 7157 if (anchor) { 7158 img.style.marginLeft = (-anchor.x) + 'px'; 7159 img.style.marginTop = (-anchor.y) + 'px'; 7160 } 7161 7162 if (size) { 7163 img.style.width = size.x + 'px'; 7164 img.style.height = size.y + 'px'; 7165 } 7166 }, 7167 7168 _createImg: function (src, el) { 7169 el = el || document.createElement('img'); 7170 el.src = src; 7171 return el; 7172 }, 7173 7174 _getIconUrl: function (name) { 7175 return retina && this.options[name + 'RetinaUrl'] || this.options[name + 'Url']; 7176 } 7177 }); 7178 7179 7180 // @factory L.icon(options: Icon options) 7181 // Creates an icon instance with the given options. 7182 function icon(options) { 7183 return new Icon(options); 7184 } 7185 7186 /* 7187 * @miniclass Icon.Default (Icon) 7188 * @aka L.Icon.Default 7189 * @section 7190 * 7191 * A trivial subclass of `Icon`, represents the icon to use in `Marker`s when 7192 * no icon is specified. Points to the blue marker image distributed with Leaflet 7193 * releases. 7194 * 7195 * In order to customize the default icon, just change the properties of `L.Icon.Default.prototype.options` 7196 * (which is a set of `Icon options`). 7197 * 7198 * If you want to _completely_ replace the default icon, override the 7199 * `L.Marker.prototype.options.icon` with your own icon instead. 7200 */ 7201 7202 var IconDefault = Icon.extend({ 7203 7204 options: { 7205 iconUrl: 'marker-icon.png', 7206 iconRetinaUrl: 'marker-icon-2x.png', 7207 shadowUrl: 'marker-shadow.png', 7208 iconSize: [25, 41], 7209 iconAnchor: [12, 41], 7210 popupAnchor: [1, -34], 7211 tooltipAnchor: [16, -28], 7212 shadowSize: [41, 41] 7213 }, 7214 7215 _getIconUrl: function (name) { 7216 if (!IconDefault.imagePath) { // Deprecated, backwards-compatibility only 7217 IconDefault.imagePath = this._detectIconPath(); 7218 } 7219 7220 // @option imagePath: String 7221 // `Icon.Default` will try to auto-detect the location of the 7222 // blue icon images. If you are placing these images in a non-standard 7223 // way, set this option to point to the right path. 7224 return (this.options.imagePath || IconDefault.imagePath) + Icon.prototype._getIconUrl.call(this, name); 7225 }, 7226 7227 _detectIconPath: function () { 7228 var el = create$1('div', 'leaflet-default-icon-path', document.body); 7229 var path = getStyle(el, 'background-image') || 7230 getStyle(el, 'backgroundImage'); // IE8 7231 7232 document.body.removeChild(el); 7233 7234 if (path === null || path.indexOf('url') !== 0) { 7235 path = ''; 7236 } else { 7237 path = path.replace(/^url\(["']?/, '').replace(/marker-icon\.png["']?\)$/, ''); 7238 } 7239 7240 return path; 7241 } 7242 }); 7243 7244 /* 7245 * L.Handler.MarkerDrag is used internally by L.Marker to make the markers draggable. 7246 */ 7247 7248 7249 /* @namespace Marker 7250 * @section Interaction handlers 7251 * 7252 * Interaction handlers are properties of a marker instance that allow you to control interaction behavior in runtime, enabling or disabling certain features such as dragging (see `Handler` methods). Example: 7253 * 7254 * ```js 7255 * marker.dragging.disable(); 7256 * ``` 7257 * 7258 * @property dragging: Handler 7259 * Marker dragging handler (by both mouse and touch). Only valid when the marker is on the map (Otherwise set [`marker.options.draggable`](#marker-draggable)). 7260 */ 7261 7262 var MarkerDrag = Handler.extend({ 7263 initialize: function (marker) { 7264 this._marker = marker; 7265 }, 7266 7267 addHooks: function () { 7268 var icon = this._marker._icon; 7269 7270 if (!this._draggable) { 7271 this._draggable = new Draggable(icon, icon, true); 7272 } 7273 7274 this._draggable.on({ 7275 dragstart: this._onDragStart, 7276 predrag: this._onPreDrag, 7277 drag: this._onDrag, 7278 dragend: this._onDragEnd 7279 }, this).enable(); 7280 7281 addClass(icon, 'leaflet-marker-draggable'); 7282 }, 7283 7284 removeHooks: function () { 7285 this._draggable.off({ 7286 dragstart: this._onDragStart, 7287 predrag: this._onPreDrag, 7288 drag: this._onDrag, 7289 dragend: this._onDragEnd 7290 }, this).disable(); 7291 7292 if (this._marker._icon) { 7293 removeClass(this._marker._icon, 'leaflet-marker-draggable'); 7294 } 7295 }, 7296 7297 moved: function () { 7298 return this._draggable && this._draggable._moved; 7299 }, 7300 7301 _adjustPan: function (e) { 7302 var marker = this._marker, 7303 map = marker._map, 7304 speed = this._marker.options.autoPanSpeed, 7305 padding = this._marker.options.autoPanPadding, 7306 iconPos = getPosition(marker._icon), 7307 bounds = map.getPixelBounds(), 7308 origin = map.getPixelOrigin(); 7309 7310 var panBounds = toBounds( 7311 bounds.min._subtract(origin).add(padding), 7312 bounds.max._subtract(origin).subtract(padding) 7313 ); 7314 7315 if (!panBounds.contains(iconPos)) { 7316 // Compute incremental movement 7317 var movement = toPoint( 7318 (Math.max(panBounds.max.x, iconPos.x) - panBounds.max.x) / (bounds.max.x - panBounds.max.x) - 7319 (Math.min(panBounds.min.x, iconPos.x) - panBounds.min.x) / (bounds.min.x - panBounds.min.x), 7320 7321 (Math.max(panBounds.max.y, iconPos.y) - panBounds.max.y) / (bounds.max.y - panBounds.max.y) - 7322 (Math.min(panBounds.min.y, iconPos.y) - panBounds.min.y) / (bounds.min.y - panBounds.min.y) 7323 ).multiplyBy(speed); 7324 7325 map.panBy(movement, {animate: false}); 7326 7327 this._draggable._newPos._add(movement); 7328 this._draggable._startPos._add(movement); 7329 7330 setPosition(marker._icon, this._draggable._newPos); 7331 this._onDrag(e); 7332 7333 this._panRequest = requestAnimFrame(this._adjustPan.bind(this, e)); 7334 } 7335 }, 7336 7337 _onDragStart: function () { 7338 // @section Dragging events 7339 // @event dragstart: Event 7340 // Fired when the user starts dragging the marker. 7341 7342 // @event movestart: Event 7343 // Fired when the marker starts moving (because of dragging). 7344 7345 this._oldLatLng = this._marker.getLatLng(); 7346 this._marker 7347 .closePopup() 7348 .fire('movestart') 7349 .fire('dragstart'); 7350 }, 7351 7352 _onPreDrag: function (e) { 7353 if (this._marker.options.autoPan) { 7354 cancelAnimFrame(this._panRequest); 7355 this._panRequest = requestAnimFrame(this._adjustPan.bind(this, e)); 7356 } 7357 }, 7358 7359 _onDrag: function (e) { 7360 var marker = this._marker, 7361 shadow = marker._shadow, 7362 iconPos = getPosition(marker._icon), 7363 latlng = marker._map.layerPointToLatLng(iconPos); 7364 7365 // update shadow position 7366 if (shadow) { 7367 setPosition(shadow, iconPos); 7368 } 7369 7370 marker._latlng = latlng; 7371 e.latlng = latlng; 7372 e.oldLatLng = this._oldLatLng; 7373 7374 // @event drag: Event 7375 // Fired repeatedly while the user drags the marker. 7376 marker 7377 .fire('move', e) 7378 .fire('drag', e); 7379 }, 7380 7381 _onDragEnd: function (e) { 7382 // @event dragend: DragEndEvent 7383 // Fired when the user stops dragging the marker. 7384 7385 cancelAnimFrame(this._panRequest); 7386 7387 // @event moveend: Event 7388 // Fired when the marker stops moving (because of dragging). 7389 delete this._oldLatLng; 7390 this._marker 7391 .fire('moveend') 7392 .fire('dragend', e); 7393 } 7394 }); 7395 7396 /* 7397 * @class Marker 7398 * @inherits Interactive layer 7399 * @aka L.Marker 7400 * L.Marker is used to display clickable/draggable icons on the map. Extends `Layer`. 7401 * 7402 * @example 7403 * 7404 * ```js 7405 * L.marker([50.5, 30.5]).addTo(map); 7406 * ``` 7407 */ 7408 7409 var Marker = Layer.extend({ 7410 7411 // @section 7412 // @aka Marker options 7413 options: { 7414 // @option icon: Icon = * 7415 // Icon instance to use for rendering the marker. 7416 // See [Icon documentation](#L.Icon) for details on how to customize the marker icon. 7417 // If not specified, a common instance of `L.Icon.Default` is used. 7418 icon: new IconDefault(), 7419 7420 // Option inherited from "Interactive layer" abstract class 7421 interactive: true, 7422 7423 // @option keyboard: Boolean = true 7424 // Whether the marker can be tabbed to with a keyboard and clicked by pressing enter. 7425 keyboard: true, 7426 7427 // @option title: String = '' 7428 // Text for the browser tooltip that appear on marker hover (no tooltip by default). 7429 title: '', 7430 7431 // @option alt: String = '' 7432 // Text for the `alt` attribute of the icon image (useful for accessibility). 7433 alt: '', 7434 7435 // @option zIndexOffset: Number = 0 7436 // By default, marker images zIndex is set automatically based on its latitude. Use this option if you want to put the marker on top of all others (or below), specifying a high value like `1000` (or high negative value, respectively). 7437 zIndexOffset: 0, 7438 7439 // @option opacity: Number = 1.0 7440 // The opacity of the marker. 7441 opacity: 1, 7442 7443 // @option riseOnHover: Boolean = false 7444 // If `true`, the marker will get on top of others when you hover the mouse over it. 7445 riseOnHover: false, 7446 7447 // @option riseOffset: Number = 250 7448 // The z-index offset used for the `riseOnHover` feature. 7449 riseOffset: 250, 7450 7451 // @option pane: String = 'markerPane' 7452 // `Map pane` where the markers icon will be added. 7453 pane: 'markerPane', 7454 7455 // @option pane: String = 'shadowPane' 7456 // `Map pane` where the markers shadow will be added. 7457 shadowPane: 'shadowPane', 7458 7459 // @option bubblingMouseEvents: Boolean = false 7460 // When `true`, a mouse event on this marker will trigger the same event on the map 7461 // (unless [`L.DomEvent.stopPropagation`](#domevent-stoppropagation) is used). 7462 bubblingMouseEvents: false, 7463 7464 // @section Draggable marker options 7465 // @option draggable: Boolean = false 7466 // Whether the marker is draggable with mouse/touch or not. 7467 draggable: false, 7468 7469 // @option autoPan: Boolean = false 7470 // Whether to pan the map when dragging this marker near its edge or not. 7471 autoPan: false, 7472 7473 // @option autoPanPadding: Point = Point(50, 50) 7474 // Distance (in pixels to the left/right and to the top/bottom) of the 7475 // map edge to start panning the map. 7476 autoPanPadding: [50, 50], 7477 7478 // @option autoPanSpeed: Number = 10 7479 // Number of pixels the map should pan by. 7480 autoPanSpeed: 10 7481 }, 7482 7483 /* @section 7484 * 7485 * In addition to [shared layer methods](#Layer) like `addTo()` and `remove()` and [popup methods](#Popup) like bindPopup() you can also use the following methods: 7486 */ 7487 7488 initialize: function (latlng, options) { 7489 setOptions(this, options); 7490 this._latlng = toLatLng(latlng); 7491 }, 7492 7493 onAdd: function (map) { 7494 this._zoomAnimated = this._zoomAnimated && map.options.markerZoomAnimation; 7495 7496 if (this._zoomAnimated) { 7497 map.on('zoomanim', this._animateZoom, this); 7498 } 7499 7500 this._initIcon(); 7501 this.update(); 7502 }, 7503 7504 onRemove: function (map) { 7505 if (this.dragging && this.dragging.enabled()) { 7506 this.options.draggable = true; 7507 this.dragging.removeHooks(); 7508 } 7509 delete this.dragging; 7510 7511 if (this._zoomAnimated) { 7512 map.off('zoomanim', this._animateZoom, this); 7513 } 7514 7515 this._removeIcon(); 7516 this._removeShadow(); 7517 }, 7518 7519 getEvents: function () { 7520 return { 7521 zoom: this.update, 7522 viewreset: this.update 7523 }; 7524 }, 7525 7526 // @method getLatLng: LatLng 7527 // Returns the current geographical position of the marker. 7528 getLatLng: function () { 7529 return this._latlng; 7530 }, 7531 7532 // @method setLatLng(latlng: LatLng): this 7533 // Changes the marker position to the given point. 7534 setLatLng: function (latlng) { 7535 var oldLatLng = this._latlng; 7536 this._latlng = toLatLng(latlng); 7537 this.update(); 7538 7539 // @event move: Event 7540 // Fired when the marker is moved via [`setLatLng`](#marker-setlatlng) or by [dragging](#marker-dragging). Old and new coordinates are included in event arguments as `oldLatLng`, `latlng`. 7541 return this.fire('move', {oldLatLng: oldLatLng, latlng: this._latlng}); 7542 }, 7543 7544 // @method setZIndexOffset(offset: Number): this 7545 // Changes the [zIndex offset](#marker-zindexoffset) of the marker. 7546 setZIndexOffset: function (offset) { 7547 this.options.zIndexOffset = offset; 7548 return this.update(); 7549 }, 7550 7551 // @method getIcon: Icon 7552 // Returns the current icon used by the marker 7553 getIcon: function () { 7554 return this.options.icon; 7555 }, 7556 7557 // @method setIcon(icon: Icon): this 7558 // Changes the marker icon. 7559 setIcon: function (icon) { 7560 7561 this.options.icon = icon; 7562 7563 if (this._map) { 7564 this._initIcon(); 7565 this.update(); 7566 } 7567 7568 if (this._popup) { 7569 this.bindPopup(this._popup, this._popup.options); 7570 } 7571 7572 return this; 7573 }, 7574 7575 getElement: function () { 7576 return this._icon; 7577 }, 7578 7579 update: function () { 7580 7581 if (this._icon && this._map) { 7582 var pos = this._map.latLngToLayerPoint(this._latlng).round(); 7583 this._setPos(pos); 7584 } 7585 7586 return this; 7587 }, 7588 7589 _initIcon: function () { 7590 var options = this.options, 7591 classToAdd = 'leaflet-zoom-' + (this._zoomAnimated ? 'animated' : 'hide'); 7592 7593 var icon = options.icon.createIcon(this._icon), 7594 addIcon = false; 7595 7596 // if we're not reusing the icon, remove the old one and init new one 7597 if (icon !== this._icon) { 7598 if (this._icon) { 7599 this._removeIcon(); 7600 } 7601 addIcon = true; 7602 7603 if (options.title) { 7604 icon.title = options.title; 7605 } 7606 7607 if (icon.tagName === 'IMG') { 7608 icon.alt = options.alt || ''; 7609 } 7610 } 7611 7612 addClass(icon, classToAdd); 7613 7614 if (options.keyboard) { 7615 icon.tabIndex = '0'; 7616 } 7617 7618 this._icon = icon; 7619 7620 if (options.riseOnHover) { 7621 this.on({ 7622 mouseover: this._bringToFront, 7623 mouseout: this._resetZIndex 7624 }); 7625 } 7626 7627 var newShadow = options.icon.createShadow(this._shadow), 7628 addShadow = false; 7629 7630 if (newShadow !== this._shadow) { 7631 this._removeShadow(); 7632 addShadow = true; 7633 } 7634 7635 if (newShadow) { 7636 addClass(newShadow, classToAdd); 7637 newShadow.alt = ''; 7638 } 7639 this._shadow = newShadow; 7640 7641 7642 if (options.opacity < 1) { 7643 this._updateOpacity(); 7644 } 7645 7646 7647 if (addIcon) { 7648 this.getPane().appendChild(this._icon); 7649 } 7650 this._initInteraction(); 7651 if (newShadow && addShadow) { 7652 this.getPane(options.shadowPane).appendChild(this._shadow); 7653 } 7654 }, 7655 7656 _removeIcon: function () { 7657 if (this.options.riseOnHover) { 7658 this.off({ 7659 mouseover: this._bringToFront, 7660 mouseout: this._resetZIndex 7661 }); 7662 } 7663 7664 remove(this._icon); 7665 this.removeInteractiveTarget(this._icon); 7666 7667 this._icon = null; 7668 }, 7669 7670 _removeShadow: function () { 7671 if (this._shadow) { 7672 remove(this._shadow); 7673 } 7674 this._shadow = null; 7675 }, 7676 7677 _setPos: function (pos) { 7678 7679 if (this._icon) { 7680 setPosition(this._icon, pos); 7681 } 7682 7683 if (this._shadow) { 7684 setPosition(this._shadow, pos); 7685 } 7686 7687 this._zIndex = pos.y + this.options.zIndexOffset; 7688 7689 this._resetZIndex(); 7690 }, 7691 7692 _updateZIndex: function (offset) { 7693 if (this._icon) { 7694 this._icon.style.zIndex = this._zIndex + offset; 7695 } 7696 }, 7697 7698 _animateZoom: function (opt) { 7699 var pos = this._map._latLngToNewLayerPoint(this._latlng, opt.zoom, opt.center).round(); 7700 7701 this._setPos(pos); 7702 }, 7703 7704 _initInteraction: function () { 7705 7706 if (!this.options.interactive) { return; } 7707 7708 addClass(this._icon, 'leaflet-interactive'); 7709 7710 this.addInteractiveTarget(this._icon); 7711 7712 if (MarkerDrag) { 7713 var draggable = this.options.draggable; 7714 if (this.dragging) { 7715 draggable = this.dragging.enabled(); 7716 this.dragging.disable(); 7717 } 7718 7719 this.dragging = new MarkerDrag(this); 7720 7721 if (draggable) { 7722 this.dragging.enable(); 7723 } 7724 } 7725 }, 7726 7727 // @method setOpacity(opacity: Number): this 7728 // Changes the opacity of the marker. 7729 setOpacity: function (opacity) { 7730 this.options.opacity = opacity; 7731 if (this._map) { 7732 this._updateOpacity(); 7733 } 7734 7735 return this; 7736 }, 7737 7738 _updateOpacity: function () { 7739 var opacity = this.options.opacity; 7740 7741 if (this._icon) { 7742 setOpacity(this._icon, opacity); 7743 } 7744 7745 if (this._shadow) { 7746 setOpacity(this._shadow, opacity); 7747 } 7748 }, 7749 7750 _bringToFront: function () { 7751 this._updateZIndex(this.options.riseOffset); 7752 }, 7753 7754 _resetZIndex: function () { 7755 this._updateZIndex(0); 7756 }, 7757 7758 _getPopupAnchor: function () { 7759 return this.options.icon.options.popupAnchor; 7760 }, 7761 7762 _getTooltipAnchor: function () { 7763 return this.options.icon.options.tooltipAnchor; 7764 } 7765 }); 7766 7767 7768 // factory L.marker(latlng: LatLng, options? : Marker options) 7769 7770 // @factory L.marker(latlng: LatLng, options? : Marker options) 7771 // Instantiates a Marker object given a geographical point and optionally an options object. 7772 function marker(latlng, options) { 7773 return new Marker(latlng, options); 7774 } 7775 7776 /* 7777 * @class Path 7778 * @aka L.Path 7779 * @inherits Interactive layer 7780 * 7781 * An abstract class that contains options and constants shared between vector 7782 * overlays (Polygon, Polyline, Circle). Do not use it directly. Extends `Layer`. 7783 */ 7784 7785 var Path = Layer.extend({ 7786 7787 // @section 7788 // @aka Path options 7789 options: { 7790 // @option stroke: Boolean = true 7791 // Whether to draw stroke along the path. Set it to `false` to disable borders on polygons or circles. 7792 stroke: true, 7793 7794 // @option color: String = '#3388ff' 7795 // Stroke color 7796 color: '#3388ff', 7797 7798 // @option weight: Number = 3 7799 // Stroke width in pixels 7800 weight: 3, 7801 7802 // @option opacity: Number = 1.0 7803 // Stroke opacity 7804 opacity: 1, 7805 7806 // @option lineCap: String= 'round' 7807 // A string that defines [shape to be used at the end](https://developer.mozilla.org/docs/Web/SVG/Attribute/stroke-linecap) of the stroke. 7808 lineCap: 'round', 7809 7810 // @option lineJoin: String = 'round' 7811 // A string that defines [shape to be used at the corners](https://developer.mozilla.org/docs/Web/SVG/Attribute/stroke-linejoin) of the stroke. 7812 lineJoin: 'round', 7813 7814 // @option dashArray: String = null 7815 // A string that defines the stroke [dash pattern](https://developer.mozilla.org/docs/Web/SVG/Attribute/stroke-dasharray). Doesn't work on `Canvas`-powered layers in [some old browsers](https://developer.mozilla.org/docs/Web/API/CanvasRenderingContext2D/setLineDash#Browser_compatibility). 7816 dashArray: null, 7817 7818 // @option dashOffset: String = null 7819 // A string that defines the [distance into the dash pattern to start the dash](https://developer.mozilla.org/docs/Web/SVG/Attribute/stroke-dashoffset). Doesn't work on `Canvas`-powered layers in [some old browsers](https://developer.mozilla.org/docs/Web/API/CanvasRenderingContext2D/setLineDash#Browser_compatibility). 7820 dashOffset: null, 7821 7822 // @option fill: Boolean = depends 7823 // Whether to fill the path with color. Set it to `false` to disable filling on polygons or circles. 7824 fill: false, 7825 7826 // @option fillColor: String = * 7827 // Fill color. Defaults to the value of the [`color`](#path-color) option 7828 fillColor: null, 7829 7830 // @option fillOpacity: Number = 0.2 7831 // Fill opacity. 7832 fillOpacity: 0.2, 7833 7834 // @option fillRule: String = 'evenodd' 7835 // A string that defines [how the inside of a shape](https://developer.mozilla.org/docs/Web/SVG/Attribute/fill-rule) is determined. 7836 fillRule: 'evenodd', 7837 7838 // className: '', 7839 7840 // Option inherited from "Interactive layer" abstract class 7841 interactive: true, 7842 7843 // @option bubblingMouseEvents: Boolean = true 7844 // When `true`, a mouse event on this path will trigger the same event on the map 7845 // (unless [`L.DomEvent.stopPropagation`](#domevent-stoppropagation) is used). 7846 bubblingMouseEvents: true 7847 }, 7848 7849 beforeAdd: function (map) { 7850 // Renderer is set here because we need to call renderer.getEvents 7851 // before this.getEvents. 7852 this._renderer = map.getRenderer(this); 7853 }, 7854 7855 onAdd: function () { 7856 this._renderer._initPath(this); 7857 this._reset(); 7858 this._renderer._addPath(this); 7859 }, 7860 7861 onRemove: function () { 7862 this._renderer._removePath(this); 7863 }, 7864 7865 // @method redraw(): this 7866 // Redraws the layer. Sometimes useful after you changed the coordinates that the path uses. 7867 redraw: function () { 7868 if (this._map) { 7869 this._renderer._updatePath(this); 7870 } 7871 return this; 7872 }, 7873 7874 // @method setStyle(style: Path options): this 7875 // Changes the appearance of a Path based on the options in the `Path options` object. 7876 setStyle: function (style) { 7877 setOptions(this, style); 7878 if (this._renderer) { 7879 this._renderer._updateStyle(this); 7880 if (this.options.stroke && style && style.hasOwnProperty('weight')) { 7881 this._updateBounds(); 7882 } 7883 } 7884 return this; 7885 }, 7886 7887 // @method bringToFront(): this 7888 // Brings the layer to the top of all path layers. 7889 bringToFront: function () { 7890 if (this._renderer) { 7891 this._renderer._bringToFront(this); 7892 } 7893 return this; 7894 }, 7895 7896 // @method bringToBack(): this 7897 // Brings the layer to the bottom of all path layers. 7898 bringToBack: function () { 7899 if (this._renderer) { 7900 this._renderer._bringToBack(this); 7901 } 7902 return this; 7903 }, 7904 7905 getElement: function () { 7906 return this._path; 7907 }, 7908 7909 _reset: function () { 7910 // defined in child classes 7911 this._project(); 7912 this._update(); 7913 }, 7914 7915 _clickTolerance: function () { 7916 // used when doing hit detection for Canvas layers 7917 return (this.options.stroke ? this.options.weight / 2 : 0) + this._renderer.options.tolerance; 7918 } 7919 }); 7920 7921 /* 7922 * @class CircleMarker 7923 * @aka L.CircleMarker 7924 * @inherits Path 7925 * 7926 * A circle of a fixed size with radius specified in pixels. Extends `Path`. 7927 */ 7928 7929 var CircleMarker = Path.extend({ 7930 7931 // @section 7932 // @aka CircleMarker options 7933 options: { 7934 fill: true, 7935 7936 // @option radius: Number = 10 7937 // Radius of the circle marker, in pixels 7938 radius: 10 7939 }, 7940 7941 initialize: function (latlng, options) { 7942 setOptions(this, options); 7943 this._latlng = toLatLng(latlng); 7944 this._radius = this.options.radius; 7945 }, 7946 7947 // @method setLatLng(latLng: LatLng): this 7948 // Sets the position of a circle marker to a new location. 7949 setLatLng: function (latlng) { 7950 var oldLatLng = this._latlng; 7951 this._latlng = toLatLng(latlng); 7952 this.redraw(); 7953 7954 // @event move: Event 7955 // Fired when the marker is moved via [`setLatLng`](#circlemarker-setlatlng). Old and new coordinates are included in event arguments as `oldLatLng`, `latlng`. 7956 return this.fire('move', {oldLatLng: oldLatLng, latlng: this._latlng}); 7957 }, 7958 7959 // @method getLatLng(): LatLng 7960 // Returns the current geographical position of the circle marker 7961 getLatLng: function () { 7962 return this._latlng; 7963 }, 7964 7965 // @method setRadius(radius: Number): this 7966 // Sets the radius of a circle marker. Units are in pixels. 7967 setRadius: function (radius) { 7968 this.options.radius = this._radius = radius; 7969 return this.redraw(); 7970 }, 7971 7972 // @method getRadius(): Number 7973 // Returns the current radius of the circle 7974 getRadius: function () { 7975 return this._radius; 7976 }, 7977 7978 setStyle : function (options) { 7979 var radius = options && options.radius || this._radius; 7980 Path.prototype.setStyle.call(this, options); 7981 this.setRadius(radius); 7982 return this; 7983 }, 7984 7985 _project: function () { 7986 this._point = this._map.latLngToLayerPoint(this._latlng); 7987 this._updateBounds(); 7988 }, 7989 7990 _updateBounds: function () { 7991 var r = this._radius, 7992 r2 = this._radiusY || r, 7993 w = this._clickTolerance(), 7994 p = [r + w, r2 + w]; 7995 this._pxBounds = new Bounds(this._point.subtract(p), this._point.add(p)); 7996 }, 7997 7998 _update: function () { 7999 if (this._map) { 8000 this._updatePath(); 8001 } 8002 }, 8003 8004 _updatePath: function () { 8005 this._renderer._updateCircle(this); 8006 }, 8007 8008 _empty: function () { 8009 return this._radius && !this._renderer._bounds.intersects(this._pxBounds); 8010 }, 8011 8012 // Needed by the `Canvas` renderer for interactivity 8013 _containsPoint: function (p) { 8014 return p.distanceTo(this._point) <= this._radius + this._clickTolerance(); 8015 } 8016 }); 8017 8018 8019 // @factory L.circleMarker(latlng: LatLng, options?: CircleMarker options) 8020 // Instantiates a circle marker object given a geographical point, and an optional options object. 8021 function circleMarker(latlng, options) { 8022 return new CircleMarker(latlng, options); 8023 } 8024 8025 /* 8026 * @class Circle 8027 * @aka L.Circle 8028 * @inherits CircleMarker 8029 * 8030 * A class for drawing circle overlays on a map. Extends `CircleMarker`. 8031 * 8032 * It's an approximation and starts to diverge from a real circle closer to poles (due to projection distortion). 8033 * 8034 * @example 8035 * 8036 * ```js 8037 * L.circle([50.5, 30.5], {radius: 200}).addTo(map); 8038 * ``` 8039 */ 8040 8041 var Circle = CircleMarker.extend({ 8042 8043 initialize: function (latlng, options, legacyOptions) { 8044 if (typeof options === 'number') { 8045 // Backwards compatibility with 0.7.x factory (latlng, radius, options?) 8046 options = extend({}, legacyOptions, {radius: options}); 8047 } 8048 setOptions(this, options); 8049 this._latlng = toLatLng(latlng); 8050 8051 if (isNaN(this.options.radius)) { throw new Error('Circle radius cannot be NaN'); } 8052 8053 // @section 8054 // @aka Circle options 8055 // @option radius: Number; Radius of the circle, in meters. 8056 this._mRadius = this.options.radius; 8057 }, 8058 8059 // @method setRadius(radius: Number): this 8060 // Sets the radius of a circle. Units are in meters. 8061 setRadius: function (radius) { 8062 this._mRadius = radius; 8063 return this.redraw(); 8064 }, 8065 8066 // @method getRadius(): Number 8067 // Returns the current radius of a circle. Units are in meters. 8068 getRadius: function () { 8069 return this._mRadius; 8070 }, 8071 8072 // @method getBounds(): LatLngBounds 8073 // Returns the `LatLngBounds` of the path. 8074 getBounds: function () { 8075 var half = [this._radius, this._radiusY || this._radius]; 8076 8077 return new LatLngBounds( 8078 this._map.layerPointToLatLng(this._point.subtract(half)), 8079 this._map.layerPointToLatLng(this._point.add(half))); 8080 }, 8081 8082 setStyle: Path.prototype.setStyle, 8083 8084 _project: function () { 8085 8086 var lng = this._latlng.lng, 8087 lat = this._latlng.lat, 8088 map = this._map, 8089 crs = map.options.crs; 8090 8091 if (crs.distance === Earth.distance) { 8092 var d = Math.PI / 180, 8093 latR = (this._mRadius / Earth.R) / d, 8094 top = map.project([lat + latR, lng]), 8095 bottom = map.project([lat - latR, lng]), 8096 p = top.add(bottom).divideBy(2), 8097 lat2 = map.unproject(p).lat, 8098 lngR = Math.acos((Math.cos(latR * d) - Math.sin(lat * d) * Math.sin(lat2 * d)) / 8099 (Math.cos(lat * d) * Math.cos(lat2 * d))) / d; 8100 8101 if (isNaN(lngR) || lngR === 0) { 8102 lngR = latR / Math.cos(Math.PI / 180 * lat); // Fallback for edge case, #2425 8103 } 8104 8105 this._point = p.subtract(map.getPixelOrigin()); 8106 this._radius = isNaN(lngR) ? 0 : p.x - map.project([lat2, lng - lngR]).x; 8107 this._radiusY = p.y - top.y; 8108 8109 } else { 8110 var latlng2 = crs.unproject(crs.project(this._latlng).subtract([this._mRadius, 0])); 8111 8112 this._point = map.latLngToLayerPoint(this._latlng); 8113 this._radius = this._point.x - map.latLngToLayerPoint(latlng2).x; 8114 } 8115 8116 this._updateBounds(); 8117 } 8118 }); 8119 8120 // @factory L.circle(latlng: LatLng, options?: Circle options) 8121 // Instantiates a circle object given a geographical point, and an options object 8122 // which contains the circle radius. 8123 // @alternative 8124 // @factory L.circle(latlng: LatLng, radius: Number, options?: Circle options) 8125 // Obsolete way of instantiating a circle, for compatibility with 0.7.x code. 8126 // Do not use in new applications or plugins. 8127 function circle(latlng, options, legacyOptions) { 8128 return new Circle(latlng, options, legacyOptions); 8129 } 8130 8131 /* 8132 * @class Polyline 8133 * @aka L.Polyline 8134 * @inherits Path 8135 * 8136 * A class for drawing polyline overlays on a map. Extends `Path`. 8137 * 8138 * @example 8139 * 8140 * ```js 8141 * // create a red polyline from an array of LatLng points 8142 * var latlngs = [ 8143 * [45.51, -122.68], 8144 * [37.77, -122.43], 8145 * [34.04, -118.2] 8146 * ]; 8147 * 8148 * var polyline = L.polyline(latlngs, {color: 'red'}).addTo(map); 8149 * 8150 * // zoom the map to the polyline 8151 * map.fitBounds(polyline.getBounds()); 8152 * ``` 8153 * 8154 * You can also pass a multi-dimensional array to represent a `MultiPolyline` shape: 8155 * 8156 * ```js 8157 * // create a red polyline from an array of arrays of LatLng points 8158 * var latlngs = [ 8159 * [[45.51, -122.68], 8160 * [37.77, -122.43], 8161 * [34.04, -118.2]], 8162 * [[40.78, -73.91], 8163 * [41.83, -87.62], 8164 * [32.76, -96.72]] 8165 * ]; 8166 * ``` 8167 */ 8168 8169 8170 var Polyline = Path.extend({ 8171 8172 // @section 8173 // @aka Polyline options 8174 options: { 8175 // @option smoothFactor: Number = 1.0 8176 // How much to simplify the polyline on each zoom level. More means 8177 // better performance and smoother look, and less means more accurate representation. 8178 smoothFactor: 1.0, 8179 8180 // @option noClip: Boolean = false 8181 // Disable polyline clipping. 8182 noClip: false 8183 }, 8184 8185 initialize: function (latlngs, options) { 8186 setOptions(this, options); 8187 this._setLatLngs(latlngs); 8188 }, 8189 8190 // @method getLatLngs(): LatLng[] 8191 // Returns an array of the points in the path, or nested arrays of points in case of multi-polyline. 8192 getLatLngs: function () { 8193 return this._latlngs; 8194 }, 8195 8196 // @method setLatLngs(latlngs: LatLng[]): this 8197 // Replaces all the points in the polyline with the given array of geographical points. 8198 setLatLngs: function (latlngs) { 8199 this._setLatLngs(latlngs); 8200 return this.redraw(); 8201 }, 8202 8203 // @method isEmpty(): Boolean 8204 // Returns `true` if the Polyline has no LatLngs. 8205 isEmpty: function () { 8206 return !this._latlngs.length; 8207 }, 8208 8209 // @method closestLayerPoint(p: Point): Point 8210 // Returns the point closest to `p` on the Polyline. 8211 closestLayerPoint: function (p) { 8212 var minDistance = Infinity, 8213 minPoint = null, 8214 closest = _sqClosestPointOnSegment, 8215 p1, p2; 8216 8217 for (var j = 0, jLen = this._parts.length; j < jLen; j++) { 8218 var points = this._parts[j]; 8219 8220 for (var i = 1, len = points.length; i < len; i++) { 8221 p1 = points[i - 1]; 8222 p2 = points[i]; 8223 8224 var sqDist = closest(p, p1, p2, true); 8225 8226 if (sqDist < minDistance) { 8227 minDistance = sqDist; 8228 minPoint = closest(p, p1, p2); 8229 } 8230 } 8231 } 8232 if (minPoint) { 8233 minPoint.distance = Math.sqrt(minDistance); 8234 } 8235 return minPoint; 8236 }, 8237 8238 // @method getCenter(): LatLng 8239 // Returns the center ([centroid](http://en.wikipedia.org/wiki/Centroid)) of the polyline. 8240 getCenter: function () { 8241 // throws error when not yet added to map as this center calculation requires projected coordinates 8242 if (!this._map) { 8243 throw new Error('Must add layer to map before using getCenter()'); 8244 } 8245 8246 var i, halfDist, segDist, dist, p1, p2, ratio, 8247 points = this._rings[0], 8248 len = points.length; 8249 8250 if (!len) { return null; } 8251 8252 // polyline centroid algorithm; only uses the first ring if there are multiple 8253 8254 for (i = 0, halfDist = 0; i < len - 1; i++) { 8255 halfDist += points[i].distanceTo(points[i + 1]) / 2; 8256 } 8257 8258 // The line is so small in the current view that all points are on the same pixel. 8259 if (halfDist === 0) { 8260 return this._map.layerPointToLatLng(points[0]); 8261 } 8262 8263 for (i = 0, dist = 0; i < len - 1; i++) { 8264 p1 = points[i]; 8265 p2 = points[i + 1]; 8266 segDist = p1.distanceTo(p2); 8267 dist += segDist; 8268 8269 if (dist > halfDist) { 8270 ratio = (dist - halfDist) / segDist; 8271 return this._map.layerPointToLatLng([ 8272 p2.x - ratio * (p2.x - p1.x), 8273 p2.y - ratio * (p2.y - p1.y) 8274 ]); 8275 } 8276 } 8277 }, 8278 8279 // @method getBounds(): LatLngBounds 8280 // Returns the `LatLngBounds` of the path. 8281 getBounds: function () { 8282 return this._bounds; 8283 }, 8284 8285 // @method addLatLng(latlng: LatLng, latlngs? LatLng[]): this 8286 // Adds a given point to the polyline. By default, adds to the first ring of 8287 // the polyline in case of a multi-polyline, but can be overridden by passing 8288 // a specific ring as a LatLng array (that you can earlier access with [`getLatLngs`](#polyline-getlatlngs)). 8289 addLatLng: function (latlng, latlngs) { 8290 latlngs = latlngs || this._defaultShape(); 8291 latlng = toLatLng(latlng); 8292 latlngs.push(latlng); 8293 this._bounds.extend(latlng); 8294 return this.redraw(); 8295 }, 8296 8297 _setLatLngs: function (latlngs) { 8298 this._bounds = new LatLngBounds(); 8299 this._latlngs = this._convertLatLngs(latlngs); 8300 }, 8301 8302 _defaultShape: function () { 8303 return isFlat(this._latlngs) ? this._latlngs : this._latlngs[0]; 8304 }, 8305 8306 // recursively convert latlngs input into actual LatLng instances; calculate bounds along the way 8307 _convertLatLngs: function (latlngs) { 8308 var result = [], 8309 flat = isFlat(latlngs); 8310 8311 for (var i = 0, len = latlngs.length; i < len; i++) { 8312 if (flat) { 8313 result[i] = toLatLng(latlngs[i]); 8314 this._bounds.extend(result[i]); 8315 } else { 8316 result[i] = this._convertLatLngs(latlngs[i]); 8317 } 8318 } 8319 8320 return result; 8321 }, 8322 8323 _project: function () { 8324 var pxBounds = new Bounds(); 8325 this._rings = []; 8326 this._projectLatlngs(this._latlngs, this._rings, pxBounds); 8327 8328 if (this._bounds.isValid() && pxBounds.isValid()) { 8329 this._rawPxBounds = pxBounds; 8330 this._updateBounds(); 8331 } 8332 }, 8333 8334 _updateBounds: function () { 8335 var w = this._clickTolerance(), 8336 p = new Point(w, w); 8337 this._pxBounds = new Bounds([ 8338 this._rawPxBounds.min.subtract(p), 8339 this._rawPxBounds.max.add(p) 8340 ]); 8341 }, 8342 8343 // recursively turns latlngs into a set of rings with projected coordinates 8344 _projectLatlngs: function (latlngs, result, projectedBounds) { 8345 var flat = latlngs[0] instanceof LatLng, 8346 len = latlngs.length, 8347 i, ring; 8348 8349 if (flat) { 8350 ring = []; 8351 for (i = 0; i < len; i++) { 8352 ring[i] = this._map.latLngToLayerPoint(latlngs[i]); 8353 projectedBounds.extend(ring[i]); 8354 } 8355 result.push(ring); 8356 } else { 8357 for (i = 0; i < len; i++) { 8358 this._projectLatlngs(latlngs[i], result, projectedBounds); 8359 } 8360 } 8361 }, 8362 8363 // clip polyline by renderer bounds so that we have less to render for performance 8364 _clipPoints: function () { 8365 var bounds = this._renderer._bounds; 8366 8367 this._parts = []; 8368 if (!this._pxBounds || !this._pxBounds.intersects(bounds)) { 8369 return; 8370 } 8371 8372 if (this.options.noClip) { 8373 this._parts = this._rings; 8374 return; 8375 } 8376 8377 var parts = this._parts, 8378 i, j, k, len, len2, segment, points; 8379 8380 for (i = 0, k = 0, len = this._rings.length; i < len; i++) { 8381 points = this._rings[i]; 8382 8383 for (j = 0, len2 = points.length; j < len2 - 1; j++) { 8384 segment = clipSegment(points[j], points[j + 1], bounds, j, true); 8385 8386 if (!segment) { continue; } 8387 8388 parts[k] = parts[k] || []; 8389 parts[k].push(segment[0]); 8390 8391 // if segment goes out of screen, or it's the last one, it's the end of the line part 8392 if ((segment[1] !== points[j + 1]) || (j === len2 - 2)) { 8393 parts[k].push(segment[1]); 8394 k++; 8395 } 8396 } 8397 } 8398 }, 8399 8400 // simplify each clipped part of the polyline for performance 8401 _simplifyPoints: function () { 8402 var parts = this._parts, 8403 tolerance = this.options.smoothFactor; 8404 8405 for (var i = 0, len = parts.length; i < len; i++) { 8406 parts[i] = simplify(parts[i], tolerance); 8407 } 8408 }, 8409 8410 _update: function () { 8411 if (!this._map) { return; } 8412 8413 this._clipPoints(); 8414 this._simplifyPoints(); 8415 this._updatePath(); 8416 }, 8417 8418 _updatePath: function () { 8419 this._renderer._updatePoly(this); 8420 }, 8421 8422 // Needed by the `Canvas` renderer for interactivity 8423 _containsPoint: function (p, closed) { 8424 var i, j, k, len, len2, part, 8425 w = this._clickTolerance(); 8426 8427 if (!this._pxBounds || !this._pxBounds.contains(p)) { return false; } 8428 8429 // hit detection for polylines 8430 for (i = 0, len = this._parts.length; i < len; i++) { 8431 part = this._parts[i]; 8432 8433 for (j = 0, len2 = part.length, k = len2 - 1; j < len2; k = j++) { 8434 if (!closed && (j === 0)) { continue; } 8435 8436 if (pointToSegmentDistance(p, part[k], part[j]) <= w) { 8437 return true; 8438 } 8439 } 8440 } 8441 return false; 8442 } 8443 }); 8444 8445 // @factory L.polyline(latlngs: LatLng[], options?: Polyline options) 8446 // Instantiates a polyline object given an array of geographical points and 8447 // optionally an options object. You can create a `Polyline` object with 8448 // multiple separate lines (`MultiPolyline`) by passing an array of arrays 8449 // of geographic points. 8450 function polyline(latlngs, options) { 8451 return new Polyline(latlngs, options); 8452 } 8453 8454 // Retrocompat. Allow plugins to support Leaflet versions before and after 1.1. 8455 Polyline._flat = _flat; 8456 8457 /* 8458 * @class Polygon 8459 * @aka L.Polygon 8460 * @inherits Polyline 8461 * 8462 * A class for drawing polygon overlays on a map. Extends `Polyline`. 8463 * 8464 * Note that points you pass when creating a polygon shouldn't have an additional last point equal to the first one — it's better to filter out such points. 8465 * 8466 * 8467 * @example 8468 * 8469 * ```js 8470 * // create a red polygon from an array of LatLng points 8471 * var latlngs = [[37, -109.05],[41, -109.03],[41, -102.05],[37, -102.04]]; 8472 * 8473 * var polygon = L.polygon(latlngs, {color: 'red'}).addTo(map); 8474 * 8475 * // zoom the map to the polygon 8476 * map.fitBounds(polygon.getBounds()); 8477 * ``` 8478 * 8479 * You can also pass an array of arrays of latlngs, with the first array representing the outer shape and the other arrays representing holes in the outer shape: 8480 * 8481 * ```js 8482 * var latlngs = [ 8483 * [[37, -109.05],[41, -109.03],[41, -102.05],[37, -102.04]], // outer ring 8484 * [[37.29, -108.58],[40.71, -108.58],[40.71, -102.50],[37.29, -102.50]] // hole 8485 * ]; 8486 * ``` 8487 * 8488 * Additionally, you can pass a multi-dimensional array to represent a MultiPolygon shape. 8489 * 8490 * ```js 8491 * var latlngs = [ 8492 * [ // first polygon 8493 * [[37, -109.05],[41, -109.03],[41, -102.05],[37, -102.04]], // outer ring 8494 * [[37.29, -108.58],[40.71, -108.58],[40.71, -102.50],[37.29, -102.50]] // hole 8495 * ], 8496 * [ // second polygon 8497 * [[41, -111.03],[45, -111.04],[45, -104.05],[41, -104.05]] 8498 * ] 8499 * ]; 8500 * ``` 8501 */ 8502 8503 var Polygon = Polyline.extend({ 8504 8505 options: { 8506 fill: true 8507 }, 8508 8509 isEmpty: function () { 8510 return !this._latlngs.length || !this._latlngs[0].length; 8511 }, 8512 8513 getCenter: function () { 8514 // throws error when not yet added to map as this center calculation requires projected coordinates 8515 if (!this._map) { 8516 throw new Error('Must add layer to map before using getCenter()'); 8517 } 8518 8519 var i, j, p1, p2, f, area, x, y, center, 8520 points = this._rings[0], 8521 len = points.length; 8522 8523 if (!len) { return null; } 8524 8525 // polygon centroid algorithm; only uses the first ring if there are multiple 8526 8527 area = x = y = 0; 8528 8529 for (i = 0, j = len - 1; i < len; j = i++) { 8530 p1 = points[i]; 8531 p2 = points[j]; 8532 8533 f = p1.y * p2.x - p2.y * p1.x; 8534 x += (p1.x + p2.x) * f; 8535 y += (p1.y + p2.y) * f; 8536 area += f * 3; 8537 } 8538 8539 if (area === 0) { 8540 // Polygon is so small that all points are on same pixel. 8541 center = points[0]; 8542 } else { 8543 center = [x / area, y / area]; 8544 } 8545 return this._map.layerPointToLatLng(center); 8546 }, 8547 8548 _convertLatLngs: function (latlngs) { 8549 var result = Polyline.prototype._convertLatLngs.call(this, latlngs), 8550 len = result.length; 8551 8552 // remove last point if it equals first one 8553 if (len >= 2 && result[0] instanceof LatLng && result[0].equals(result[len - 1])) { 8554 result.pop(); 8555 } 8556 return result; 8557 }, 8558 8559 _setLatLngs: function (latlngs) { 8560 Polyline.prototype._setLatLngs.call(this, latlngs); 8561 if (isFlat(this._latlngs)) { 8562 this._latlngs = [this._latlngs]; 8563 } 8564 }, 8565 8566 _defaultShape: function () { 8567 return isFlat(this._latlngs[0]) ? this._latlngs[0] : this._latlngs[0][0]; 8568 }, 8569 8570 _clipPoints: function () { 8571 // polygons need a different clipping algorithm so we redefine that 8572 8573 var bounds = this._renderer._bounds, 8574 w = this.options.weight, 8575 p = new Point(w, w); 8576 8577 // increase clip padding by stroke width to avoid stroke on clip edges 8578 bounds = new Bounds(bounds.min.subtract(p), bounds.max.add(p)); 8579 8580 this._parts = []; 8581 if (!this._pxBounds || !this._pxBounds.intersects(bounds)) { 8582 return; 8583 } 8584 8585 if (this.options.noClip) { 8586 this._parts = this._rings; 8587 return; 8588 } 8589 8590 for (var i = 0, len = this._rings.length, clipped; i < len; i++) { 8591 clipped = clipPolygon(this._rings[i], bounds, true); 8592 if (clipped.length) { 8593 this._parts.push(clipped); 8594 } 8595 } 8596 }, 8597 8598 _updatePath: function () { 8599 this._renderer._updatePoly(this, true); 8600 }, 8601 8602 // Needed by the `Canvas` renderer for interactivity 8603 _containsPoint: function (p) { 8604 var inside = false, 8605 part, p1, p2, i, j, k, len, len2; 8606 8607 if (!this._pxBounds || !this._pxBounds.contains(p)) { return false; } 8608 8609 // ray casting algorithm for detecting if point is in polygon 8610 for (i = 0, len = this._parts.length; i < len; i++) { 8611 part = this._parts[i]; 8612 8613 for (j = 0, len2 = part.length, k = len2 - 1; j < len2; k = j++) { 8614 p1 = part[j]; 8615 p2 = part[k]; 8616 8617 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)) { 8618 inside = !inside; 8619 } 8620 } 8621 } 8622 8623 // also check if it's on polygon stroke 8624 return inside || Polyline.prototype._containsPoint.call(this, p, true); 8625 } 8626 8627 }); 8628 8629 8630 // @factory L.polygon(latlngs: LatLng[], options?: Polyline options) 8631 function polygon(latlngs, options) { 8632 return new Polygon(latlngs, options); 8633 } 8634 8635 /* 8636 * @class GeoJSON 8637 * @aka L.GeoJSON 8638 * @inherits FeatureGroup 8639 * 8640 * Represents a GeoJSON object or an array of GeoJSON objects. Allows you to parse 8641 * GeoJSON data and display it on the map. Extends `FeatureGroup`. 8642 * 8643 * @example 8644 * 8645 * ```js 8646 * L.geoJSON(data, { 8647 * style: function (feature) { 8648 * return {color: feature.properties.color}; 8649 * } 8650 * }).bindPopup(function (layer) { 8651 * return layer.feature.properties.description; 8652 * }).addTo(map); 8653 * ``` 8654 */ 8655 8656 var GeoJSON = FeatureGroup.extend({ 8657 8658 /* @section 8659 * @aka GeoJSON options 8660 * 8661 * @option pointToLayer: Function = * 8662 * A `Function` defining how GeoJSON points spawn Leaflet layers. It is internally 8663 * called when data is added, passing the GeoJSON point feature and its `LatLng`. 8664 * The default is to spawn a default `Marker`: 8665 * ```js 8666 * function(geoJsonPoint, latlng) { 8667 * return L.marker(latlng); 8668 * } 8669 * ``` 8670 * 8671 * @option style: Function = * 8672 * A `Function` defining the `Path options` for styling GeoJSON lines and polygons, 8673 * called internally when data is added. 8674 * The default value is to not override any defaults: 8675 * ```js 8676 * function (geoJsonFeature) { 8677 * return {} 8678 * } 8679 * ``` 8680 * 8681 * @option onEachFeature: Function = * 8682 * A `Function` that will be called once for each created `Feature`, after it has 8683 * been created and styled. Useful for attaching events and popups to features. 8684 * The default is to do nothing with the newly created layers: 8685 * ```js 8686 * function (feature, layer) {} 8687 * ``` 8688 * 8689 * @option filter: Function = * 8690 * A `Function` that will be used to decide whether to include a feature or not. 8691 * The default is to include all features: 8692 * ```js 8693 * function (geoJsonFeature) { 8694 * return true; 8695 * } 8696 * ``` 8697 * Note: dynamically changing the `filter` option will have effect only on newly 8698 * added data. It will _not_ re-evaluate already included features. 8699 * 8700 * @option coordsToLatLng: Function = * 8701 * A `Function` that will be used for converting GeoJSON coordinates to `LatLng`s. 8702 * The default is the `coordsToLatLng` static method. 8703 * 8704 * @option markersInheritOptions: Boolean = false 8705 * Whether default Markers for "Point" type Features inherit from group options. 8706 */ 8707 8708 initialize: function (geojson, options) { 8709 setOptions(this, options); 8710 8711 this._layers = {}; 8712 8713 if (geojson) { 8714 this.addData(geojson); 8715 } 8716 }, 8717 8718 // @method addData( <GeoJSON> data ): this 8719 // Adds a GeoJSON object to the layer. 8720 addData: function (geojson) { 8721 var features = isArray(geojson) ? geojson : geojson.features, 8722 i, len, feature; 8723 8724 if (features) { 8725 for (i = 0, len = features.length; i < len; i++) { 8726 // only add this if geometry or geometries are set and not null 8727 feature = features[i]; 8728 if (feature.geometries || feature.geometry || feature.features || feature.coordinates) { 8729 this.addData(feature); 8730 } 8731 } 8732 return this; 8733 } 8734 8735 var options = this.options; 8736 8737 if (options.filter && !options.filter(geojson)) { return this; } 8738 8739 var layer = geometryToLayer(geojson, options); 8740 if (!layer) { 8741 return this; 8742 } 8743 layer.feature = asFeature(geojson); 8744 8745 layer.defaultOptions = layer.options; 8746 this.resetStyle(layer); 8747 8748 if (options.onEachFeature) { 8749 options.onEachFeature(geojson, layer); 8750 } 8751 8752 return this.addLayer(layer); 8753 }, 8754 8755 // @method resetStyle( <Path> layer? ): this 8756 // Resets the given vector layer's style to the original GeoJSON style, useful for resetting style after hover events. 8757 // If `layer` is omitted, the style of all features in the current layer is reset. 8758 resetStyle: function (layer) { 8759 if (layer === undefined) { 8760 return this.eachLayer(this.resetStyle, this); 8761 } 8762 // reset any custom styles 8763 layer.options = extend({}, layer.defaultOptions); 8764 this._setLayerStyle(layer, this.options.style); 8765 return this; 8766 }, 8767 8768 // @method setStyle( <Function> style ): this 8769 // Changes styles of GeoJSON vector layers with the given style function. 8770 setStyle: function (style) { 8771 return this.eachLayer(function (layer) { 8772 this._setLayerStyle(layer, style); 8773 }, this); 8774 }, 8775 8776 _setLayerStyle: function (layer, style) { 8777 if (layer.setStyle) { 8778 if (typeof style === 'function') { 8779 style = style(layer.feature); 8780 } 8781 layer.setStyle(style); 8782 } 8783 } 8784 }); 8785 8786 // @section 8787 // There are several static functions which can be called without instantiating L.GeoJSON: 8788 8789 // @function geometryToLayer(featureData: Object, options?: GeoJSON options): Layer 8790 // Creates a `Layer` from a given GeoJSON feature. Can use a custom 8791 // [`pointToLayer`](#geojson-pointtolayer) and/or [`coordsToLatLng`](#geojson-coordstolatlng) 8792 // functions if provided as options. 8793 function geometryToLayer(geojson, options) { 8794 8795 var geometry = geojson.type === 'Feature' ? geojson.geometry : geojson, 8796 coords = geometry ? geometry.coordinates : null, 8797 layers = [], 8798 pointToLayer = options && options.pointToLayer, 8799 _coordsToLatLng = options && options.coordsToLatLng || coordsToLatLng, 8800 latlng, latlngs, i, len; 8801 8802 if (!coords && !geometry) { 8803 return null; 8804 } 8805 8806 switch (geometry.type) { 8807 case 'Point': 8808 latlng = _coordsToLatLng(coords); 8809 return _pointToLayer(pointToLayer, geojson, latlng, options); 8810 8811 case 'MultiPoint': 8812 for (i = 0, len = coords.length; i < len; i++) { 8813 latlng = _coordsToLatLng(coords[i]); 8814 layers.push(_pointToLayer(pointToLayer, geojson, latlng, options)); 8815 } 8816 return new FeatureGroup(layers); 8817 8818 case 'LineString': 8819 case 'MultiLineString': 8820 latlngs = coordsToLatLngs(coords, geometry.type === 'LineString' ? 0 : 1, _coordsToLatLng); 8821 return new Polyline(latlngs, options); 8822 8823 case 'Polygon': 8824 case 'MultiPolygon': 8825 latlngs = coordsToLatLngs(coords, geometry.type === 'Polygon' ? 1 : 2, _coordsToLatLng); 8826 return new Polygon(latlngs, options); 8827 8828 case 'GeometryCollection': 8829 for (i = 0, len = geometry.geometries.length; i < len; i++) { 8830 var layer = geometryToLayer({ 8831 geometry: geometry.geometries[i], 8832 type: 'Feature', 8833 properties: geojson.properties 8834 }, options); 8835 8836 if (layer) { 8837 layers.push(layer); 8838 } 8839 } 8840 return new FeatureGroup(layers); 8841 8842 default: 8843 throw new Error('Invalid GeoJSON object.'); 8844 } 8845 } 8846 8847 function _pointToLayer(pointToLayerFn, geojson, latlng, options) { 8848 return pointToLayerFn ? 8849 pointToLayerFn(geojson, latlng) : 8850 new Marker(latlng, options && options.markersInheritOptions && options); 8851 } 8852 8853 // @function coordsToLatLng(coords: Array): LatLng 8854 // Creates a `LatLng` object from an array of 2 numbers (longitude, latitude) 8855 // or 3 numbers (longitude, latitude, altitude) used in GeoJSON for points. 8856 function coordsToLatLng(coords) { 8857 return new LatLng(coords[1], coords[0], coords[2]); 8858 } 8859 8860 // @function coordsToLatLngs(coords: Array, levelsDeep?: Number, coordsToLatLng?: Function): Array 8861 // Creates a multidimensional array of `LatLng`s from a GeoJSON coordinates array. 8862 // `levelsDeep` specifies the nesting level (0 is for an array of points, 1 for an array of arrays of points, etc., 0 by default). 8863 // Can use a custom [`coordsToLatLng`](#geojson-coordstolatlng) function. 8864 function coordsToLatLngs(coords, levelsDeep, _coordsToLatLng) { 8865 var latlngs = []; 8866 8867 for (var i = 0, len = coords.length, latlng; i < len; i++) { 8868 latlng = levelsDeep ? 8869 coordsToLatLngs(coords[i], levelsDeep - 1, _coordsToLatLng) : 8870 (_coordsToLatLng || coordsToLatLng)(coords[i]); 8871 8872 latlngs.push(latlng); 8873 } 8874 8875 return latlngs; 8876 } 8877 8878 // @function latLngToCoords(latlng: LatLng, precision?: Number): Array 8879 // Reverse of [`coordsToLatLng`](#geojson-coordstolatlng) 8880 function latLngToCoords(latlng, precision) { 8881 precision = typeof precision === 'number' ? precision : 6; 8882 return latlng.alt !== undefined ? 8883 [formatNum(latlng.lng, precision), formatNum(latlng.lat, precision), formatNum(latlng.alt, precision)] : 8884 [formatNum(latlng.lng, precision), formatNum(latlng.lat, precision)]; 8885 } 8886 8887 // @function latLngsToCoords(latlngs: Array, levelsDeep?: Number, closed?: Boolean): Array 8888 // Reverse of [`coordsToLatLngs`](#geojson-coordstolatlngs) 8889 // `closed` determines whether the first point should be appended to the end of the array to close the feature, only used when `levelsDeep` is 0. False by default. 8890 function latLngsToCoords(latlngs, levelsDeep, closed, precision) { 8891 var coords = []; 8892 8893 for (var i = 0, len = latlngs.length; i < len; i++) { 8894 coords.push(levelsDeep ? 8895 latLngsToCoords(latlngs[i], levelsDeep - 1, closed, precision) : 8896 latLngToCoords(latlngs[i], precision)); 8897 } 8898 8899 if (!levelsDeep && closed) { 8900 coords.push(coords[0]); 8901 } 8902 8903 return coords; 8904 } 8905 8906 function getFeature(layer, newGeometry) { 8907 return layer.feature ? 8908 extend({}, layer.feature, {geometry: newGeometry}) : 8909 asFeature(newGeometry); 8910 } 8911 8912 // @function asFeature(geojson: Object): Object 8913 // Normalize GeoJSON geometries/features into GeoJSON features. 8914 function asFeature(geojson) { 8915 if (geojson.type === 'Feature' || geojson.type === 'FeatureCollection') { 8916 return geojson; 8917 } 8918 8919 return { 8920 type: 'Feature', 8921 properties: {}, 8922 geometry: geojson 8923 }; 8924 } 8925 8926 var PointToGeoJSON = { 8927 toGeoJSON: function (precision) { 8928 return getFeature(this, { 8929 type: 'Point', 8930 coordinates: latLngToCoords(this.getLatLng(), precision) 8931 }); 8932 } 8933 }; 8934 8935 // @namespace Marker 8936 // @section Other methods 8937 // @method toGeoJSON(precision?: Number): Object 8938 // `precision` is the number of decimal places for coordinates. 8939 // The default value is 6 places. 8940 // Returns a [`GeoJSON`](http://en.wikipedia.org/wiki/GeoJSON) representation of the marker (as a GeoJSON `Point` Feature). 8941 Marker.include(PointToGeoJSON); 8942 8943 // @namespace CircleMarker 8944 // @method toGeoJSON(precision?: Number): Object 8945 // `precision` is the number of decimal places for coordinates. 8946 // The default value is 6 places. 8947 // Returns a [`GeoJSON`](http://en.wikipedia.org/wiki/GeoJSON) representation of the circle marker (as a GeoJSON `Point` Feature). 8948 Circle.include(PointToGeoJSON); 8949 CircleMarker.include(PointToGeoJSON); 8950 8951 8952 // @namespace Polyline 8953 // @method toGeoJSON(precision?: Number): Object 8954 // `precision` is the number of decimal places for coordinates. 8955 // The default value is 6 places. 8956 // Returns a [`GeoJSON`](http://en.wikipedia.org/wiki/GeoJSON) representation of the polyline (as a GeoJSON `LineString` or `MultiLineString` Feature). 8957 Polyline.include({ 8958 toGeoJSON: function (precision) { 8959 var multi = !isFlat(this._latlngs); 8960 8961 var coords = latLngsToCoords(this._latlngs, multi ? 1 : 0, false, precision); 8962 8963 return getFeature(this, { 8964 type: (multi ? 'Multi' : '') + 'LineString', 8965 coordinates: coords 8966 }); 8967 } 8968 }); 8969 8970 // @namespace Polygon 8971 // @method toGeoJSON(precision?: Number): Object 8972 // `precision` is the number of decimal places for coordinates. 8973 // The default value is 6 places. 8974 // Returns a [`GeoJSON`](http://en.wikipedia.org/wiki/GeoJSON) representation of the polygon (as a GeoJSON `Polygon` or `MultiPolygon` Feature). 8975 Polygon.include({ 8976 toGeoJSON: function (precision) { 8977 var holes = !isFlat(this._latlngs), 8978 multi = holes && !isFlat(this._latlngs[0]); 8979 8980 var coords = latLngsToCoords(this._latlngs, multi ? 2 : holes ? 1 : 0, true, precision); 8981 8982 if (!holes) { 8983 coords = [coords]; 8984 } 8985 8986 return getFeature(this, { 8987 type: (multi ? 'Multi' : '') + 'Polygon', 8988 coordinates: coords 8989 }); 8990 } 8991 }); 8992 8993 8994 // @namespace LayerGroup 8995 LayerGroup.include({ 8996 toMultiPoint: function (precision) { 8997 var coords = []; 8998 8999 this.eachLayer(function (layer) { 9000 coords.push(layer.toGeoJSON(precision).geometry.coordinates); 9001 }); 9002 9003 return getFeature(this, { 9004 type: 'MultiPoint', 9005 coordinates: coords 9006 }); 9007 }, 9008 9009 // @method toGeoJSON(precision?: Number): Object 9010 // `precision` is the number of decimal places for coordinates. 9011 // The default value is 6 places. 9012 // Returns a [`GeoJSON`](http://en.wikipedia.org/wiki/GeoJSON) representation of the layer group (as a GeoJSON `FeatureCollection`, `GeometryCollection`, or `MultiPoint`). 9013 toGeoJSON: function (precision) { 9014 9015 var type = this.feature && this.feature.geometry && this.feature.geometry.type; 9016 9017 if (type === 'MultiPoint') { 9018 return this.toMultiPoint(precision); 9019 } 9020 9021 var isGeometryCollection = type === 'GeometryCollection', 9022 jsons = []; 9023 9024 this.eachLayer(function (layer) { 9025 if (layer.toGeoJSON) { 9026 var json = layer.toGeoJSON(precision); 9027 if (isGeometryCollection) { 9028 jsons.push(json.geometry); 9029 } else { 9030 var feature = asFeature(json); 9031 // Squash nested feature collections 9032 if (feature.type === 'FeatureCollection') { 9033 jsons.push.apply(jsons, feature.features); 9034 } else { 9035 jsons.push(feature); 9036 } 9037 } 9038 } 9039 }); 9040 9041 if (isGeometryCollection) { 9042 return getFeature(this, { 9043 geometries: jsons, 9044 type: 'GeometryCollection' 9045 }); 9046 } 9047 9048 return { 9049 type: 'FeatureCollection', 9050 features: jsons 9051 }; 9052 } 9053 }); 9054 9055 // @namespace GeoJSON 9056 // @factory L.geoJSON(geojson?: Object, options?: GeoJSON options) 9057 // Creates a GeoJSON layer. Optionally accepts an object in 9058 // [GeoJSON format](https://tools.ietf.org/html/rfc7946) to display on the map 9059 // (you can alternatively add it later with `addData` method) and an `options` object. 9060 function geoJSON(geojson, options) { 9061 return new GeoJSON(geojson, options); 9062 } 9063 9064 // Backward compatibility. 9065 var geoJson = geoJSON; 9066 9067 /* 9068 * @class ImageOverlay 9069 * @aka L.ImageOverlay 9070 * @inherits Interactive layer 9071 * 9072 * Used to load and display a single image over specific bounds of the map. Extends `Layer`. 9073 * 9074 * @example 9075 * 9076 * ```js 9077 * var imageUrl = 'http://www.lib.utexas.edu/maps/historical/newark_nj_1922.jpg', 9078 * imageBounds = [[40.712216, -74.22655], [40.773941, -74.12544]]; 9079 * L.imageOverlay(imageUrl, imageBounds).addTo(map); 9080 * ``` 9081 */ 9082 9083 var ImageOverlay = Layer.extend({ 9084 9085 // @section 9086 // @aka ImageOverlay options 9087 options: { 9088 // @option opacity: Number = 1.0 9089 // The opacity of the image overlay. 9090 opacity: 1, 9091 9092 // @option alt: String = '' 9093 // Text for the `alt` attribute of the image (useful for accessibility). 9094 alt: '', 9095 9096 // @option interactive: Boolean = false 9097 // If `true`, the image overlay will emit [mouse events](#interactive-layer) when clicked or hovered. 9098 interactive: false, 9099 9100 // @option crossOrigin: Boolean|String = false 9101 // Whether the crossOrigin attribute will be added to the image. 9102 // If a String is provided, the image will have its crossOrigin attribute set to the String provided. This is needed if you want to access image pixel data. 9103 // Refer to [CORS Settings](https://developer.mozilla.org/en-US/docs/Web/HTML/CORS_settings_attributes) for valid String values. 9104 crossOrigin: false, 9105 9106 // @option errorOverlayUrl: String = '' 9107 // URL to the overlay image to show in place of the overlay that failed to load. 9108 errorOverlayUrl: '', 9109 9110 // @option zIndex: Number = 1 9111 // The explicit [zIndex](https://developer.mozilla.org/docs/Web/CSS/CSS_Positioning/Understanding_z_index) of the overlay layer. 9112 zIndex: 1, 9113 9114 // @option className: String = '' 9115 // A custom class name to assign to the image. Empty by default. 9116 className: '' 9117 }, 9118 9119 initialize: function (url, bounds, options) { // (String, LatLngBounds, Object) 9120 this._url = url; 9121 this._bounds = toLatLngBounds(bounds); 9122 9123 setOptions(this, options); 9124 }, 9125 9126 onAdd: function () { 9127 if (!this._image) { 9128 this._initImage(); 9129 9130 if (this.options.opacity < 1) { 9131 this._updateOpacity(); 9132 } 9133 } 9134 9135 if (this.options.interactive) { 9136 addClass(this._image, 'leaflet-interactive'); 9137 this.addInteractiveTarget(this._image); 9138 } 9139 9140 this.getPane().appendChild(this._image); 9141 this._reset(); 9142 }, 9143 9144 onRemove: function () { 9145 remove(this._image); 9146 if (this.options.interactive) { 9147 this.removeInteractiveTarget(this._image); 9148 } 9149 }, 9150 9151 // @method setOpacity(opacity: Number): this 9152 // Sets the opacity of the overlay. 9153 setOpacity: function (opacity) { 9154 this.options.opacity = opacity; 9155 9156 if (this._image) { 9157 this._updateOpacity(); 9158 } 9159 return this; 9160 }, 9161 9162 setStyle: function (styleOpts) { 9163 if (styleOpts.opacity) { 9164 this.setOpacity(styleOpts.opacity); 9165 } 9166 return this; 9167 }, 9168 9169 // @method bringToFront(): this 9170 // Brings the layer to the top of all overlays. 9171 bringToFront: function () { 9172 if (this._map) { 9173 toFront(this._image); 9174 } 9175 return this; 9176 }, 9177 9178 // @method bringToBack(): this 9179 // Brings the layer to the bottom of all overlays. 9180 bringToBack: function () { 9181 if (this._map) { 9182 toBack(this._image); 9183 } 9184 return this; 9185 }, 9186 9187 // @method setUrl(url: String): this 9188 // Changes the URL of the image. 9189 setUrl: function (url) { 9190 this._url = url; 9191 9192 if (this._image) { 9193 this._image.src = url; 9194 } 9195 return this; 9196 }, 9197 9198 // @method setBounds(bounds: LatLngBounds): this 9199 // Update the bounds that this ImageOverlay covers 9200 setBounds: function (bounds) { 9201 this._bounds = toLatLngBounds(bounds); 9202 9203 if (this._map) { 9204 this._reset(); 9205 } 9206 return this; 9207 }, 9208 9209 getEvents: function () { 9210 var events = { 9211 zoom: this._reset, 9212 viewreset: this._reset 9213 }; 9214 9215 if (this._zoomAnimated) { 9216 events.zoomanim = this._animateZoom; 9217 } 9218 9219 return events; 9220 }, 9221 9222 // @method setZIndex(value: Number): this 9223 // Changes the [zIndex](#imageoverlay-zindex) of the image overlay. 9224 setZIndex: function (value) { 9225 this.options.zIndex = value; 9226 this._updateZIndex(); 9227 return this; 9228 }, 9229 9230 // @method getBounds(): LatLngBounds 9231 // Get the bounds that this ImageOverlay covers 9232 getBounds: function () { 9233 return this._bounds; 9234 }, 9235 9236 // @method getElement(): HTMLElement 9237 // Returns the instance of [`HTMLImageElement`](https://developer.mozilla.org/docs/Web/API/HTMLImageElement) 9238 // used by this overlay. 9239 getElement: function () { 9240 return this._image; 9241 }, 9242 9243 _initImage: function () { 9244 var wasElementSupplied = this._url.tagName === 'IMG'; 9245 var img = this._image = wasElementSupplied ? this._url : create$1('img'); 9246 9247 addClass(img, 'leaflet-image-layer'); 9248 if (this._zoomAnimated) { addClass(img, 'leaflet-zoom-animated'); } 9249 if (this.options.className) { addClass(img, this.options.className); } 9250 9251 img.onselectstart = falseFn; 9252 img.onmousemove = falseFn; 9253 9254 // @event load: Event 9255 // Fired when the ImageOverlay layer has loaded its image 9256 img.onload = bind(this.fire, this, 'load'); 9257 img.onerror = bind(this._overlayOnError, this, 'error'); 9258 9259 if (this.options.crossOrigin || this.options.crossOrigin === '') { 9260 img.crossOrigin = this.options.crossOrigin === true ? '' : this.options.crossOrigin; 9261 } 9262 9263 if (this.options.zIndex) { 9264 this._updateZIndex(); 9265 } 9266 9267 if (wasElementSupplied) { 9268 this._url = img.src; 9269 return; 9270 } 9271 9272 img.src = this._url; 9273 img.alt = this.options.alt; 9274 }, 9275 9276 _animateZoom: function (e) { 9277 var scale = this._map.getZoomScale(e.zoom), 9278 offset = this._map._latLngBoundsToNewLayerBounds(this._bounds, e.zoom, e.center).min; 9279 9280 setTransform(this._image, offset, scale); 9281 }, 9282 9283 _reset: function () { 9284 var image = this._image, 9285 bounds = new Bounds( 9286 this._map.latLngToLayerPoint(this._bounds.getNorthWest()), 9287 this._map.latLngToLayerPoint(this._bounds.getSouthEast())), 9288 size = bounds.getSize(); 9289 9290 setPosition(image, bounds.min); 9291 9292 image.style.width = size.x + 'px'; 9293 image.style.height = size.y + 'px'; 9294 }, 9295 9296 _updateOpacity: function () { 9297 setOpacity(this._image, this.options.opacity); 9298 }, 9299 9300 _updateZIndex: function () { 9301 if (this._image && this.options.zIndex !== undefined && this.options.zIndex !== null) { 9302 this._image.style.zIndex = this.options.zIndex; 9303 } 9304 }, 9305 9306 _overlayOnError: function () { 9307 // @event error: Event 9308 // Fired when the ImageOverlay layer fails to load its image 9309 this.fire('error'); 9310 9311 var errorUrl = this.options.errorOverlayUrl; 9312 if (errorUrl && this._url !== errorUrl) { 9313 this._url = errorUrl; 9314 this._image.src = errorUrl; 9315 } 9316 } 9317 }); 9318 9319 // @factory L.imageOverlay(imageUrl: String, bounds: LatLngBounds, options?: ImageOverlay options) 9320 // Instantiates an image overlay object given the URL of the image and the 9321 // geographical bounds it is tied to. 9322 var imageOverlay = function (url, bounds, options) { 9323 return new ImageOverlay(url, bounds, options); 9324 }; 9325 9326 /* 9327 * @class VideoOverlay 9328 * @aka L.VideoOverlay 9329 * @inherits ImageOverlay 9330 * 9331 * Used to load and display a video player over specific bounds of the map. Extends `ImageOverlay`. 9332 * 9333 * A video overlay uses the [`<video>`](https://developer.mozilla.org/docs/Web/HTML/Element/video) 9334 * HTML5 element. 9335 * 9336 * @example 9337 * 9338 * ```js 9339 * var videoUrl = 'https://www.mapbox.com/bites/00188/patricia_nasa.webm', 9340 * videoBounds = [[ 32, -130], [ 13, -100]]; 9341 * L.videoOverlay(videoUrl, videoBounds ).addTo(map); 9342 * ``` 9343 */ 9344 9345 var VideoOverlay = ImageOverlay.extend({ 9346 9347 // @section 9348 // @aka VideoOverlay options 9349 options: { 9350 // @option autoplay: Boolean = true 9351 // Whether the video starts playing automatically when loaded. 9352 autoplay: true, 9353 9354 // @option loop: Boolean = true 9355 // Whether the video will loop back to the beginning when played. 9356 loop: true, 9357 9358 // @option keepAspectRatio: Boolean = true 9359 // Whether the video will save aspect ratio after the projection. 9360 // Relevant for supported browsers. Browser compatibility- https://developer.mozilla.org/en-US/docs/Web/CSS/object-fit 9361 keepAspectRatio: true 9362 }, 9363 9364 _initImage: function () { 9365 var wasElementSupplied = this._url.tagName === 'VIDEO'; 9366 var vid = this._image = wasElementSupplied ? this._url : create$1('video'); 9367 9368 addClass(vid, 'leaflet-image-layer'); 9369 if (this._zoomAnimated) { addClass(vid, 'leaflet-zoom-animated'); } 9370 if (this.options.className) { addClass(vid, this.options.className); } 9371 9372 vid.onselectstart = falseFn; 9373 vid.onmousemove = falseFn; 9374 9375 // @event load: Event 9376 // Fired when the video has finished loading the first frame 9377 vid.onloadeddata = bind(this.fire, this, 'load'); 9378 9379 if (wasElementSupplied) { 9380 var sourceElements = vid.getElementsByTagName('source'); 9381 var sources = []; 9382 for (var j = 0; j < sourceElements.length; j++) { 9383 sources.push(sourceElements[j].src); 9384 } 9385 9386 this._url = (sourceElements.length > 0) ? sources : [vid.src]; 9387 return; 9388 } 9389 9390 if (!isArray(this._url)) { this._url = [this._url]; } 9391 9392 if (!this.options.keepAspectRatio && vid.style.hasOwnProperty('objectFit')) { vid.style['objectFit'] = 'fill'; } 9393 vid.autoplay = !!this.options.autoplay; 9394 vid.loop = !!this.options.loop; 9395 for (var i = 0; i < this._url.length; i++) { 9396 var source = create$1('source'); 9397 source.src = this._url[i]; 9398 vid.appendChild(source); 9399 } 9400 } 9401 9402 // @method getElement(): HTMLVideoElement 9403 // Returns the instance of [`HTMLVideoElement`](https://developer.mozilla.org/docs/Web/API/HTMLVideoElement) 9404 // used by this overlay. 9405 }); 9406 9407 9408 // @factory L.videoOverlay(video: String|Array|HTMLVideoElement, bounds: LatLngBounds, options?: VideoOverlay options) 9409 // Instantiates an image overlay object given the URL of the video (or array of URLs, or even a video element) and the 9410 // geographical bounds it is tied to. 9411 9412 function videoOverlay(video, bounds, options) { 9413 return new VideoOverlay(video, bounds, options); 9414 } 9415 9416 /* 9417 * @class SVGOverlay 9418 * @aka L.SVGOverlay 9419 * @inherits ImageOverlay