/** * Wraps a Leaflet map in an {@link Ext.Component} using the [Leaflet API](http://leafletjs.com/reference.html). * * To use this component you must include an additional JavaScript an a CSS file from Leaflet: * * <link rel="stylesheet" type="text/css" href="http://cdn.leafletjs.com/leaflet-0.6.4/leaflet.css"> * <script type="text/javascript" src="http://cdn.leafletjs.com/leaflet-0.6.4/leaflet.js"></script> * * ## Example * * Ext.Viewport.add({ * xtype: 'leafletmap', * useCurrentLocation: true * }); * */ Ext.define('Ext.ux.LeafletMap', { extend: 'Ext.Container', xtype: 'leafletmap', requires: ['Ext.util.Geolocation'], config: { /** * @event maprender * Fired when map is initially rendered. * @param {Ext.ux.LeafletMap} this * @param {L.Map} map The rendered L.Map instance * @param {L.TileLayer} tileLayer The rendered L.TileLayer instance */ /** * @event zoomend * Fired when a map zoom ended. * @param {Ext.ux.LeafletMap} this * @param {L.Map} map The rendered L.Map instance * @param {L.TileLayer} tileLayer The rendered L.TileLayer instance * @param {Number} zoom The current zoom level of the map */ /** * @event movestart * Fired when a panning on map starts. * @param {Ext.ux.LeafletMap} this * @param {L.Map} map The rendered L.Map instance * @param {L.TileLayer} tileLayer The rendered L.TileLayer instance */ /** * @event moveend * Fired when a panning on map ends. * @param {Ext.ux.LeafletMap} this * @param {L.Map} map The rendered L.Map instance * @param {L.TileLayer} tileLayer The rendered L.TileLayer instance */ /** * @cfg {String} baseCls * The base CSS class to apply to the map's element * @accessor */ baseCls: Ext.baseCSSPrefix + 'llmap', /** * @cfg {Boolean/Ext.util.Geolocation} useCurrentLocation * Pass in true to center the map based on the geolocation coordinates or pass a * {@link Ext.util.Geolocation GeoLocation} config to have more control over your GeoLocation options * @accessor */ useCurrentLocation: false, /** * @cfg {L.Map} map * The wrapped map. * @accessor */ map: null, /** * @cfg {Object} mapOptions * MapOptions as specified by the Leaflet documentation: * [http://leafletjs.com/reference.html#map-class](http://leafletjs.com/reference.html#map-class) * @accessor */ mapOptions: {}, /** * @cfg {String} [tileLayerUrl="http://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png"] * URL template for tile-layer in the following form * * 'http://{s}.somedomain.com/blabla/{z}/{x}/{y}.png' * * {s} means one of the randomly chosen subdomains (their range is specified in options; a, b or c by default, * can be omitted), {z} — zoom level, {x} and {y} — tile coordinates. * * You can use custom keys in the template, which will be evaluated from {@link Ext.ux.LeafletMap#tileLayerOptions}, like this: * * tileLayerUrl: 'http://{s}.somedomain.com/{foo}/{z}/{x}/{y}.png', tileLayerOptions: {foo: 'bar'}; * * @accessor */ tileLayerUrl: 'http://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', /** * @cfg {Object} tileLayerOptions * Tile-layer options which should be used in the L.TileLayer constructor. * @accessor */ tileLayerOptions: {}, /** * @cfg {String} [retinaTileLayerUrl="http://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png"] * URL template for tile-layer with retina tiles in the following form * * 'http://{s}.somedomain.com/style@2x/{z}/{x}/{y}.png' * * {s} means one of the randomly chosen subdomains (their range is specified in options; a, b or c by default, * can be omitted), {z} — zoom level, {x} and {y} — tile coordinates. * * If retina url is defined it's used if the device has a retina display. * * You can use custom keys in the template, which will be evaluated from {@link Ext.ux.LeafletMap#tileLayerOptions}, like this: * * retinaTileLayerUrl: 'http://{s}.somedomain.com/{foo}@2x/{z}/{x}/{y}.png', tileLayerOptions: {foo: 'bar'}; * * @accessor */ retinaTileLayerUrl: null, /** * @cfg {L.TileLayer} tileLayer * The wrapped layer. * @accessor */ tileLayer: null, /** * @cfg {Ext.util.Geolocation} geo * Geolocation provider for the map. * @accessor */ geo: null, /** * @cfg {Boolean} autoMapCenter * Defines if the map should automatically center itself on a geoupdate event. * Only applies if {@link Ext.ux.LeafletMap#useCurrentLocation} is set to true. * @accessor */ autoMapCenter: false, /** * @cfg {Boolean} initialCenter * Defines if the map initially should be centered to the current location. * @accessor */ initialCenter: true, /** * @cfg {Boolean} enableOwnPositionMarker * Defines if a marker should be placed on the current position. * This marker automatically updates its position on a location update event. * Only works if useCurrentLocation is set to true. * @accessor */ enableOwnPositionMarker: false, /** * @cfg {L.Marker} ownPositionMarker * Marker object which shows the current location. * @accessor */ ownPositionMarker: null, /** * @cfg {Object} ownPositionMarkerIcon * Options for the icon of own position marker. * See [L.Icon](http://leafletjs.com/reference.html#icon) documentation for possible options. * @accessor */ ownPositionMarkerIcon: { iconUrl: '', iconSize: [20, 20], iconAnchor: [20/2, 20/2] }, /** * @cfg {Object} ownPositionMarkerOptions * Own position marker options. * See [L.Marker](http://leafletjs.com/reference.html#marker) documentation for possible options. * @accessor */ ownPositionMarkerOptions: { clickable: false } }, constructor: function () { this.callParent(arguments); var ll = window.L; if (!ll) { this.setHtml('Leaflet library is required'); } }, initialize: function () { this.callParent(); this.on({ painted: 'doResize', scope: this }); this.innerElement.on('touchstart', 'onTouchStart', this); }, getElementConfig: function () { return { reference: 'element', className: 'x-container', children: [{ reference: 'innerElement', className: 'x-inner', children: [{ reference: 'mapContainer', className: Ext.baseCSSPrefix + 'map-container' }] }] }; }, onTouchStart: function (e) { e.makeUnpreventable(); }, applyMapOptions: function (options) { return Ext.merge({}, this.options, options); }, updateMapOptions: function (newOptions) { var me = this, ll = window.L, map = this.getMap(); if (ll && map) { map.setOptions(newOptions); } if (newOptions.center && !me.isPainted()) { me.un('painted', 'setMapCenter', this); me.on('painted', 'setMapCenter', this, { delay: 150, single: true, args: [newOptions.center] }); } }, getMapOptions: function () { return Ext.merge({}, this.options || this.getInitialConfig('mapOptions')); }, getTileLayerOptions: function () { return Ext.merge({}, this.options || this.getInitialConfig('tileLayerOptions')); }, updateUseCurrentLocation: function (useCurrentLocation) { this.setGeo(useCurrentLocation); if (!this.getMap() && (!useCurrentLocation || !this.getInitialCenter())) { this.renderMap(); } }, applyGeo: function (config) { return Ext.factory(config, Ext.util.Geolocation, this.getGeo()); }, updateGeo: function (newGeo, oldGeo) { var events = { locationupdate : 'onGeoUpdate', locationerror : 'onGeoError', scope : this }; if (oldGeo) { oldGeo.un(events); } if (newGeo) { newGeo.on(events); newGeo.updateLocation(); } }, doResize: function () { var ll = window.L, map = this.getMap(); if (ll && map) { map.invalidateSize(); } }, // @private renderMap: function () { var me = this, ll = window.L, element = me.mapContainer, mapOptions = me.getMapOptions(), map, tileLayer; // if map isn't painted yet -> recall method after a certain time if (!me.isPainted()) { me.un('painted', 'renderMap', this); me.on('painted', 'renderMap', this, { delay: 150, single: true, args: [] }); return; } if (ll && !element.dom._leaflet) { // if no center property is given -> use default position if (!mapOptions.hasOwnProperty('center')) { mapOptions.center = new ll.LatLng(47.36865, 8.539183); // default: Zuerich } if (mapOptions.center && mapOptions.center.lat && mapOptions.center.lng) { mapOptions.center = new ll.LatLng(mapOptions.center.lat, mapOptions.center.lng); } if(me.getRetinaTileLayerUrl() !== null && me.getRetinaTileLayerUrl() !== "" && ll.Browser.retina) { me.setTileLayer(new ll.TileLayer(me.getRetinaTileLayerUrl(), me.getTileLayerOptions())); } else { me.setTileLayer(new ll.TileLayer(me.getTileLayerUrl(), me.getTileLayerOptions())); } tileLayer = me.getTileLayer(); mapOptions.layers = [tileLayer]; me.setMap(new ll.Map(element.dom, mapOptions)); map = me.getMap(); // add own position marker if enabled if(me.getGeo() && me.getEnableOwnPositionMarker()) { me.addOwnPositionMarker(); } // track map events map.on('zoomend', me.onZoomEnd, me); map.on('movestart', me.onMoveStart, me); map.on('moveend', me.onMoveEnd, me); me.fireEvent('maprender', me, map, tileLayer); } }, // @private onGeoUpdate: function (geo) { var ll = window.L, ownPositionMarker = this.getOwnPositionMarker(); if (ll && geo && (this.getAutoMapCenter() || this.getInitialCenter())) { this.setMapCenter(new ll.LatLng(geo.getLatitude(), geo.getLongitude())); this.setInitialCenter(false); } if(ownPositionMarker) { ownPositionMarker.setLatLng(ll.latLng(geo.getLatitude(), geo.getLongitude())); } }, // @private onGeoError: function (geo) { this.setUseCurrentLocation(false); if(!this.getMap()) { this.renderMap(); } }, /** * Moves the map center to the designated coordinates hash of the form: * * { latitude: 47.36865, longitude: 8.539183 } * * or a L.LatLng object representing to the target location. * * @param {Object/L.LatLng} coordinates Object representing the desired longitude and * latitude upon which to center the map. */ setMapCenter: function (coordinates) { var me = this, map = me.getMap(), ll = window.L; if (ll) { if (!me.isPainted()) { me.un('painted', 'setMapCenter', this); me.on('painted', 'setMapCenter', this, { delay: 150, single: true, args: [coordinates] }); return; } coordinates = coordinates || new ll.LatLng(47.36865, 8.539183); if (coordinates && !(coordinates instanceof ll.LatLng) && coordinates.hasOwnProperty('latitude')) { coordinates = new ll.LatLng(coordinates.latitude, coordinates.longitude); } if (!map) { me.renderMap(); map = me.getMap(); } if (map && coordinates instanceof ll.LatLng) { map.panTo(coordinates); } else { this.options = Ext.apply(this.getMapOptions(), { center: coordinates }); } } }, /** * @private * Adds own position marker to map */ addOwnPositionMarker: function() { var me = this, ll = window.L, icon, iconOptions, ownPositionMarker, markerOptions; iconOptions = Ext.merge({}, me.getOwnPositionMarkerIcon()); icon = ll.icon(iconOptions); markerOptions = Ext.merge({ icon: icon }, me.getOwnPositionMarkerOptions()); ownPositionMarker = ll.marker([me.getGeo().getLatitude(), me.getGeo().getLongitude()], markerOptions); me.setOwnPositionMarker(ownPositionMarker); ownPositionMarker.addTo(me.getMap()); }, // @private onZoomEnd: function () { var mapOptions = this.getMapOptions(), map = this.getMap(), tileLayer = this.getTileLayer(), zoom; zoom = map.getZoom() || 10; this.options = Ext.apply(mapOptions, { zoom: zoom }); this.fireEvent('zoomend', this, map, tileLayer, zoom); }, // @private onMoveStart: function () { var map = this.getMap(), tileLayer = this.getTileLayer(); this.fireEvent('movestart', this, map, tileLayer); }, // @private onMoveEnd: function () { var map = this.getMap(), tileLayer = this.getTileLayer(); this.fireEvent('moveend', this, map, tileLayer); }, // @private destroy: function () { Ext.destroy(this.getGeo()); var map = this.getMap(), layer = this.getTileLayer(); if (map) { map = null; } if (layer) { layer = null; } this.callParent(); } });