Switched to vue-leaflet implementation for web client.
							
								
								
									
										23
									
								
								cmd/mtwebmapper/client/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,23 @@ | ||||
| .DS_Store | ||||
| node_modules | ||||
| /dist | ||||
|  | ||||
|  | ||||
| # local env files | ||||
| .env.local | ||||
| .env.*.local | ||||
|  | ||||
| # Log files | ||||
| npm-debug.log* | ||||
| yarn-debug.log* | ||||
| yarn-error.log* | ||||
| pnpm-debug.log* | ||||
|  | ||||
| # Editor directories and files | ||||
| .idea | ||||
| .vscode | ||||
| *.suo | ||||
| *.ntvs* | ||||
| *.njsproj | ||||
| *.sln | ||||
| *.sw? | ||||
							
								
								
									
										24
									
								
								cmd/mtwebmapper/client/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,24 @@ | ||||
| # client | ||||
|  | ||||
| ## Project setup | ||||
| ``` | ||||
| yarn install | ||||
| ``` | ||||
|  | ||||
| ### Compiles and hot-reloads for development | ||||
| ``` | ||||
| yarn serve | ||||
| ``` | ||||
|  | ||||
| ### Compiles and minifies for production | ||||
| ``` | ||||
| yarn build | ||||
| ``` | ||||
|  | ||||
| ### Lints and fixes files | ||||
| ``` | ||||
| yarn lint | ||||
| ``` | ||||
|  | ||||
| ### Customize configuration | ||||
| See [Configuration Reference](https://cli.vuejs.org/config/). | ||||
							
								
								
									
										5
									
								
								cmd/mtwebmapper/client/babel.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,5 @@ | ||||
| module.exports = { | ||||
|   "presets": [ | ||||
|     "@vue/cli-plugin-babel/preset" | ||||
|   ] | ||||
| } | ||||
							
								
								
									
										19
									
								
								cmd/mtwebmapper/client/jsconfig.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,19 @@ | ||||
| { | ||||
|   "compilerOptions": { | ||||
|     "target": "es5", | ||||
|     "module": "esnext", | ||||
|     "baseUrl": "./", | ||||
|     "moduleResolution": "node", | ||||
|     "paths": { | ||||
|       "@/*": [ | ||||
|         "src/*" | ||||
|       ] | ||||
|     }, | ||||
|     "lib": [ | ||||
|       "esnext", | ||||
|       "dom", | ||||
|       "dom.iterable", | ||||
|       "scripthost" | ||||
|     ] | ||||
|   } | ||||
| } | ||||
							
								
								
									
										54
									
								
								cmd/mtwebmapper/client/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,54 @@ | ||||
| { | ||||
|   "name": "client", | ||||
|   "version": "0.1.0", | ||||
|   "private": true, | ||||
|   "scripts": { | ||||
|     "serve": "vue-cli-service serve", | ||||
|     "build": "vue-cli-service build", | ||||
|     "lint": "vue-cli-service lint" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "@babel/polyfill": "^7.4.4", | ||||
|     "core-js": "^3.8.3", | ||||
|     "leaflet": "^1.7.1", | ||||
|     "leaflet.coordinates": "^0.1.5", | ||||
|     "vue": "^2.6.14", | ||||
|     "vue2-leaflet": "^2.7.1", | ||||
|     "vue2-leaflet-markercluster": "^3.1.0", | ||||
|     "vuetify": "^2.6.0", | ||||
|     "vuex": "^3.6.2" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@babel/core": "^7.12.16", | ||||
|     "@babel/eslint-parser": "^7.12.16", | ||||
|     "@fortawesome/fontawesome-free": "^6.0.0", | ||||
|     "@vue/cli-plugin-babel": "~5.0.0", | ||||
|     "@vue/cli-plugin-eslint": "~5.0.0", | ||||
|     "@vue/cli-plugin-vuex": "~5.0.0", | ||||
|     "@vue/cli-service": "~5.0.0", | ||||
|     "eslint": "^7.32.0", | ||||
|     "eslint-plugin-vue": "^8.0.3", | ||||
|     "vue-cli-plugin-vuetify": "~2.4.6", | ||||
|     "vue-template-compiler": "^2.6.14" | ||||
|   }, | ||||
|   "eslintConfig": { | ||||
|     "root": true, | ||||
|     "env": { | ||||
|       "node": true | ||||
|     }, | ||||
|     "extends": [ | ||||
|       "plugin:vue/essential", | ||||
|       "eslint:recommended" | ||||
|     ], | ||||
|     "parserOptions": { | ||||
|       "parser": "@babel/eslint-parser" | ||||
|     }, | ||||
|     "rules": {} | ||||
|   }, | ||||
|   "browserslist": [ | ||||
|     "> 1%", | ||||
|     "last 2 versions", | ||||
|     "not dead", | ||||
|     "not ie <= 10" | ||||
|   ] | ||||
| } | ||||
							
								
								
									
										
											BIN
										
									
								
								cmd/mtwebmapper/client/public/favicon.ico
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 4.2 KiB | 
							
								
								
									
										18
									
								
								cmd/mtwebmapper/client/public/index.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,18 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
|   <head> | ||||
|     <meta charset="utf-8"> | ||||
|     <meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
|     <meta name="viewport" content="width=device-width,initial-scale=1.0"> | ||||
|     <link rel="icon" href="<%= BASE_URL %>favicon.ico"> | ||||
|     <title><%= htmlWebpackPlugin.options.title %></title> | ||||
|     <link rel="stylesheet" href="https://fonts.googleapis.com/css?family=Roboto:100,300,400,500,700,900"> | ||||
|   </head> | ||||
|   <body> | ||||
|     <noscript> | ||||
|       <strong>We're sorry but <%= htmlWebpackPlugin.options.title %> doesn't work properly without JavaScript enabled. Please enable it to continue.</strong> | ||||
|     </noscript> | ||||
|     <div id="app"></div> | ||||
|     <!-- built files will be auto injected --> | ||||
|   </body> | ||||
| </html> | ||||
							
								
								
									
										147
									
								
								cmd/mtwebmapper/client/src/App.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,147 @@ | ||||
| <template> | ||||
|   <v-app dark> | ||||
|     <v-navigation-drawer | ||||
|       v-model="drawer" | ||||
|       :clipped="true" | ||||
|       fixed | ||||
|       app | ||||
|     > | ||||
|       <v-list> | ||||
|         <v-list-item> | ||||
|           <v-list-item-action> | ||||
|             <v-checkbox v-model="playersLayer"></v-checkbox> | ||||
|           </v-list-item-action> | ||||
|           <v-list-item-content> | ||||
|             <v-list-item-title>Players</v-list-item-title> | ||||
|           </v-list-item-content> | ||||
|         </v-list-item> | ||||
|       </v-list> | ||||
|     </v-navigation-drawer> | ||||
|     <v-app-bar :clipped-left="true" fixed app> | ||||
|       <v-app-bar-nav-icon @click.stop="drawer = !drawer" /> | ||||
|       <v-toolbar-title v-text="title" /> | ||||
|       <v-spacer /> | ||||
|       <v-btn icon @click.stop="refreshContent"> | ||||
|         <v-icon v-if="!autoUpdate">fa-refresh</v-icon> | ||||
|       </v-btn> | ||||
|       <v-btn icon @click.stop="autoUpdate = !autoUpdate"> | ||||
|         <v-icon v-if="!autoUpdate">fa-play</v-icon> | ||||
|         <v-icon v-else>fa-pause</v-icon> | ||||
|       </v-btn> | ||||
|     </v-app-bar> | ||||
|     <v-main> | ||||
|         <MapView /> | ||||
|     </v-main> | ||||
|     <v-footer :absolute="!fixed" app> | ||||
|       <span>© {{ new Date().getFullYear() }}</span> | ||||
|       <v-spacer /> | ||||
|       <span>X: {{coordinates[0]}} - Y: {{coordinates[1]}}</span> | ||||
|     </v-footer> | ||||
|   </v-app> | ||||
| </template> | ||||
|  | ||||
| <script> | ||||
| import MapView from './components/MapView.vue' | ||||
| import {mapState} from 'vuex'; | ||||
| export default { | ||||
|   name: 'App', | ||||
|   components: { | ||||
|     MapView | ||||
|   }, | ||||
|   data() { | ||||
|     return { | ||||
|       drawer: false, | ||||
|       fixed: false, | ||||
|       layerItems: [ | ||||
|         { | ||||
|           icon: 'mdi-apps', | ||||
|           title: 'Points of Interest', | ||||
|         }, | ||||
|         { | ||||
|           icon: 'mdi-chart-bubble', | ||||
|           title: 'Buildings', | ||||
|         }, | ||||
|       ], | ||||
|       title: 'Minetest - Map of the Real World', | ||||
|     } | ||||
|   }, | ||||
|   computed: { | ||||
|     ...mapState({ | ||||
|       players: state => state.players, | ||||
|     }), | ||||
|     autoUpdate: { | ||||
|       get() { | ||||
|         return this.$store.state.autoUpdate; | ||||
|       }, | ||||
|       set(value) { | ||||
|         if (value) { | ||||
|           this.$store.dispatch("runAutoUpdate") | ||||
|         } | ||||
|         else { | ||||
|           this.$store.dispatch("stopAutoUpdate") | ||||
|         } | ||||
|         this.$store.commit("setAutoUpdate", value); | ||||
|       } | ||||
|     }, | ||||
|     coordinates() { | ||||
|       return this.$store.state.coordinates; | ||||
|     }, | ||||
|     playersLayer: { | ||||
|       get() { | ||||
|         return this.$store.state.playersLayer; | ||||
|       }, | ||||
|       set(value) { | ||||
|         this.$store.commit("updatePlayersLayer", value) | ||||
|       }  | ||||
|     }, | ||||
|     buildingsLayer: { | ||||
|       get() { | ||||
|         return this.$store.state.buildingsLayer; | ||||
|       }, | ||||
|       set(value) { | ||||
|         this.$store.commit("updateBuildingsLayer", value) | ||||
|       }  | ||||
|     }, | ||||
|     poiLayer: { | ||||
|       get() { | ||||
|         return this.$store.state.poiLayer; | ||||
|       }, | ||||
|       set(value) { | ||||
|         this.$store.commit("updatePoiLayer", value) | ||||
|       }  | ||||
|     }, | ||||
|     travelnetLayer: { | ||||
|       get() { | ||||
|         return this.$store.state.travelnetLayer; | ||||
|       }, | ||||
|       set(value) { | ||||
|         this.$store.commit("updateTravelnetLayer", value) | ||||
|       }  | ||||
|     }, | ||||
|     otherLayer: { | ||||
|       get() { | ||||
|         return this.$store.state.otherLayer; | ||||
|       }, | ||||
|       set(value) { | ||||
|         this.$store.commit("updateOtherLayer", value) | ||||
|       }  | ||||
|     } | ||||
|   }, | ||||
|   mounted() { | ||||
|     fetch("players").then(response => response.json().then((data) => { | ||||
|       this.$store.commit("setPlayers", data); | ||||
|     })) | ||||
|   }, | ||||
|   methods: { | ||||
|     refreshContent() { | ||||
|       this.$store.dispatch("manualUpdate"); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <style> | ||||
| .v-navigation-drawer .v-navigation-drawer--fixed{ | ||||
|     z-index: 1001; | ||||
| } | ||||
| </style> | ||||
							
								
								
									
										
											BIN
										
									
								
								cmd/mtwebmapper/client/src/assets/logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 6.7 KiB | 
							
								
								
									
										
											BIN
										
									
								
								cmd/mtwebmapper/client/src/assets/marker-player.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 12 KiB | 
							
								
								
									
										84
									
								
								cmd/mtwebmapper/client/src/components/MapView.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,84 @@ | ||||
| <template> | ||||
|   <v-container fluid style="height: 100%; padding: 0px"> | ||||
|     <l-map @mousemove="update" ref="map" style="height: 100%" :zoom="zoom" :center="center" :worldCopyJump="worldCopyJump" :crs="crs" :options="mapOptions"> | ||||
|       <l-tile-layer ref="baseLayer" :url="url" :attribution="attribution" :tms="tms" continuousWorld="false" minZoom=0 :noWrap="noWrap" :options="layerOptions"></l-tile-layer> | ||||
|       <l-marker v-for="p in players" :key="p.name" :lat-lng="flipCoordinates(p.geometry.coordinates)" :icon="iconPlayer"> | ||||
|         <l-popup>{{p.properties.name}}</l-popup> | ||||
|       </l-marker> | ||||
|     </l-map> | ||||
|   </v-container> | ||||
| </template> | ||||
|  | ||||
| <script> | ||||
| import L from 'leaflet'; | ||||
| import {mapState} from 'vuex'; | ||||
|  | ||||
| L.Projection.NoWrap = { | ||||
|   project(latlng) { | ||||
|       return new L.Point(latlng.lat, latlng.lng); | ||||
|   }, | ||||
|   unproject(point) { | ||||
|       return new L.LatLng(point.x, point.y, true); | ||||
|   }, | ||||
|   bounds: L.bounds(L.point(-30928.0,30928.0),L.point(30928.0, -30928.0))  | ||||
| }; | ||||
| const ownCRS = L.CRS.Direct = L.Util.extend({}, L.CRS, { | ||||
|   code: 'Direct', | ||||
|   projection: L.Projection.NoWrap, | ||||
|   transformation: new L.Transformation(1.0/65536, 30928.0/65536, -1.0/65536, 34608.0/65536) | ||||
| }); | ||||
|  | ||||
| export default { | ||||
|   name: 'MapView', | ||||
|   data: () => { | ||||
|     return { | ||||
|       mapOptions: { | ||||
|         fadeAnimation: false | ||||
|       }, | ||||
|       layerOptions: { | ||||
|         unloadInvisibleTiles: true | ||||
|       }, | ||||
|       socket: null, | ||||
|       url: 'map/{z}/{x}/{y}.png', | ||||
|       zoom: 3, | ||||
|       noWrap: true, | ||||
|       center: [0,0], | ||||
|       worldCopyJump: false, | ||||
|       crs: ownCRS, | ||||
|       tms: true, | ||||
|       attribution: 'Demo World', | ||||
|       features: [], | ||||
|       iconPlayer: new L.Icon({ | ||||
|         iconUrl: require('@/assets/marker-player.png'), | ||||
|         iconSize: [25, 41], | ||||
|         iconAnchor: [12, 41], | ||||
|         popupAnchor: [1, -34], | ||||
|       }), | ||||
|     } | ||||
|   }, | ||||
|   computed: { | ||||
|     players() { | ||||
|       return this.$store.state.players; | ||||
|     }, | ||||
|     ...mapState({ | ||||
|       autoUpdate: state => state.autoUpdate, | ||||
|       playersLayer: state => state.playersLayer, | ||||
|     }), | ||||
|   }, | ||||
|   methods: { | ||||
|     update(evt) { | ||||
|       this.$store.commit('updateCoordinates', [evt.latlng.lat, evt.latlng.lng]) | ||||
|     }, | ||||
|     flipCoordinates(pos) { | ||||
|       return [pos[1], pos[0]]; | ||||
|     } | ||||
|   }, | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <style> | ||||
| .leaflet-touch .leaflet-bar a { | ||||
|   color: #f5f5f5; | ||||
|   background-color: #424242; | ||||
| } | ||||
| </style> | ||||
							
								
								
									
										14
									
								
								cmd/mtwebmapper/client/src/main.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,14 @@ | ||||
| import Vue from 'vue' | ||||
| import App from './App.vue' | ||||
| import vuetify from './plugins/vuetify' | ||||
| import './plugins/leaflet' | ||||
| import '@babel/polyfill' | ||||
| import store from './store' | ||||
|  | ||||
| Vue.config.productionTip = false | ||||
|  | ||||
| new Vue({ | ||||
|   vuetify, | ||||
|   store, | ||||
|   render: h => h(App) | ||||
| }).$mount('#app') | ||||
							
								
								
									
										12
									
								
								cmd/mtwebmapper/client/src/plugins/leaflet.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | ||||
| import Vue from 'vue'; | ||||
| import { LMap, LTileLayer, LMarker, LPopup } from 'vue2-leaflet'; | ||||
| import Vue2LeafletMarkerCluster from 'vue2-leaflet-markercluster' | ||||
| import 'leaflet/dist/leaflet.css'; | ||||
| import "leaflet.markercluster/dist/MarkerCluster.css"; | ||||
| import "leaflet.markercluster/dist/MarkerCluster.Default.css"; | ||||
|  | ||||
| Vue.component('l-map', LMap); | ||||
| Vue.component('l-tile-layer', LTileLayer); | ||||
| Vue.component('l-marker', LMarker); | ||||
| Vue.component('l-popup', LPopup); | ||||
| Vue.component('l-marker-cluster', Vue2LeafletMarkerCluster); | ||||
							
								
								
									
										27
									
								
								cmd/mtwebmapper/client/src/plugins/vuetify.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,27 @@ | ||||
| import Vue from 'vue'; | ||||
| import Vuetify from 'vuetify'; | ||||
| import 'vuetify/dist/vuetify.min.css'; | ||||
| import colors from 'vuetify/es5/util/colors' | ||||
| import '@fortawesome/fontawesome-free/css/all.css' | ||||
|  | ||||
| Vue.use(Vuetify); | ||||
|  | ||||
| export default new Vuetify({ | ||||
|   icons: { | ||||
|     iconfont: 'fa', | ||||
|   }, | ||||
|   theme: { | ||||
|     dark: true, | ||||
|     themes: { | ||||
|       dark: { | ||||
|         primary: colors.blue.darken2, | ||||
|         accent: colors.grey.darken3, | ||||
|         secondary: colors.amber.darken3, | ||||
|         info: colors.teal.lighten1, | ||||
|         warning: colors.amber.base, | ||||
|         error: colors.deepOrange.accent4, | ||||
|         success: colors.green.accent3, | ||||
|       }, | ||||
|     }, | ||||
|   }, | ||||
| }); | ||||
							
								
								
									
										189
									
								
								cmd/mtwebmapper/client/src/store/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,189 @@ | ||||
| import Vue from 'vue' | ||||
| import Vuex from 'vuex' | ||||
|  | ||||
| Vue.use(Vuex) | ||||
|  | ||||
| export default new Vuex.Store({ | ||||
| 	state: { | ||||
| 		socket: null, | ||||
| 		autoUpdate: false, | ||||
| 		coordinates: [0, 0], | ||||
| 		playersLayer: true, | ||||
| 		buildingsLayer: true, | ||||
| 		poiLayer: true, | ||||
| 		travelnetLayer: true, | ||||
| 		otherLayer: true, | ||||
| 		players: [] | ||||
| 	}, | ||||
| 	 | ||||
| 	mutations: { | ||||
| 		updateCoordinates(state, value) { | ||||
| 			state.coordinates = value; | ||||
| 		}, | ||||
| 		setAutoUpdate(state, value) { | ||||
| 			state.autoUpdate = value | ||||
| 		}, | ||||
| 		updatePlayersLayer(state, value) { | ||||
| 			state.playersLayer = value; | ||||
| 		}, | ||||
| 		updateBuildingsLayer(state, value) { | ||||
| 			state.buildingsLayer = value; | ||||
| 		}, | ||||
| 		updatePoiLayer(state, value) { | ||||
| 			state.poiLayer = value; | ||||
| 		}, | ||||
| 		updateTravelnetLayer(state, value) { | ||||
| 			state.travelnetLayer = value; | ||||
| 		}, | ||||
| 		updateOtherLayer(state, value) { | ||||
| 			state.otherLayer = value; | ||||
| 		}, | ||||
| 		setPlayers(state, value) { | ||||
| 			state.players = value; | ||||
| 		}, | ||||
| 		setPlayerState(state, value) { | ||||
| 			const p = state.players.find(p => p.name === value.name) | ||||
| 			p.online = value.state; | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	actions: { | ||||
| 		stopAutoUpdate: function({ state }) { | ||||
| 			if (this.socket) { | ||||
| 				var s = state.socket; | ||||
| 				state.socket = null; | ||||
| 				s.close(); | ||||
| 			} | ||||
| 		}, | ||||
|  | ||||
| 		manualUpdate: function({commit}) { | ||||
|         var tiles = document.getElementsByTagName("img"); | ||||
|         for (var i = 0; i < tiles.length; i++) { | ||||
|             var img = tiles[i]; | ||||
|             var cl = img.getAttribute("class"); | ||||
|             if (cl.indexOf("leaflet-tile-loaded") >= 0) { | ||||
|                 var src = img.src; | ||||
|                 var idx = src.lastIndexOf("#"); | ||||
|                 if (idx >= 0) { | ||||
|                     src = src.substring(0, idx); | ||||
|                 } | ||||
|                 img.src = src + "#" + Math.random(); | ||||
|             } | ||||
|         } | ||||
| 				fetch("players").then(response => response.json().then((data) => { | ||||
| 					commit("setPlayers", data); | ||||
| 				})); | ||||
|     }, | ||||
|  | ||||
| 		runAutoUpdate: function({ commit, state, dispatch}) { | ||||
| 			var me = this; | ||||
| 			// TODO: Make the URL to websocket configurable via .env | ||||
| 			state.socket = new WebSocket('ws://' + window.location.host + '/socket'); | ||||
|  | ||||
| 			state.socket.onerror = function() { | ||||
| 				commit('setAutoUpdate', false) | ||||
| 				dispatch('stopAutoUpdate'); | ||||
| 			}; | ||||
|  | ||||
| 			state.socket.onclose = function() { | ||||
| 				commit('setAutoUpdate', false) | ||||
| 				state.socket = null; | ||||
| 			} | ||||
|  | ||||
| 			state.socket.onopen = function() { | ||||
| 				// Sending pings every 5 secs to keep connection alive. | ||||
| 				var heartbeat = function() { | ||||
| 					if (heartbeat && me.socket) { | ||||
| 						me.socket.send("PING"); | ||||
| 						setTimeout(heartbeat, 8000); | ||||
| 					} else { | ||||
| 						// Prevent sending pings to re-opened sockets. | ||||
| 						heartbeat = null; | ||||
| 					} | ||||
| 				}; | ||||
| 				setTimeout(heartbeat, 8000); | ||||
| 			}; | ||||
|  | ||||
| 			state.socket.onmessage = function(evt) { | ||||
| 				var json = evt.data; | ||||
| 				if (!(typeof json === "string")) { | ||||
| 					return; | ||||
| 				} | ||||
|  | ||||
| 				var msg; | ||||
| 				try { | ||||
| 					msg = JSON.parse(json); | ||||
| 				} | ||||
| 				catch (err) { | ||||
| 					return; | ||||
| 				} | ||||
|  | ||||
| 				if (msg.players) { | ||||
| 					commit('setPlayers', msg.players); | ||||
| 				} | ||||
|  | ||||
| 				var tilesData = msg.tiles; | ||||
| 				if (!tilesData) { | ||||
| 					return; | ||||
| 				} | ||||
|  | ||||
| 				var invalidate = function(td) { | ||||
| 					var pyramid = new Array(9); | ||||
| 					var last = new Object(); | ||||
| 					pyramid[8] = last; | ||||
|  | ||||
| 					for (var i = 0; i < td.length; i++) { | ||||
| 							var xz = td[i]; | ||||
| 							last[xz.X + "#" + xz.Z] = xz; | ||||
| 					} | ||||
| 					for (var p = 7; p >= 0; p--) { | ||||
| 						var prev = pyramid[p+1]; | ||||
| 						var curr = new Object(); | ||||
| 						pyramid[p] = curr; | ||||
| 						for (var k in prev) { | ||||
| 							if (Object.prototype.hasOwnProperty.call(prev, k)) { | ||||
| 								var oxz = prev[k]; | ||||
| 								var nxz = { X: oxz.X >> 1, Z: oxz.Z >> 1 }; | ||||
| 								curr[nxz.X + "#" + nxz.Z] = nxz; | ||||
| 							} | ||||
| 						} | ||||
| 					} | ||||
| 					return function(x, y, z) { | ||||
| 						if (y > 8) { | ||||
| 							x >>= y - 8; | ||||
| 							z >>= y - 8; | ||||
| 							y = 8; | ||||
| 						} | ||||
| 						var level = pyramid[y]; | ||||
| 						var k = x + "#" + z; | ||||
| 						return Object.prototype.hasOwnProperty.call(level, k); | ||||
| 					}; | ||||
| 				} (tilesData); | ||||
| 				var tiles = document.getElementsByTagName('img'); | ||||
| 				var re = /\/map\/([0-9]+)\/([0-9]+)\/([0-9]+).*/; | ||||
| 				for (var i = 0; i < tiles.length; i++) { | ||||
| 					var img = tiles[i]; | ||||
| 					var cl = img.getAttribute('class'); | ||||
| 					if (cl.indexOf('leaflet-tile-loaded') < 0) { | ||||
| 						continue; | ||||
| 					} | ||||
| 					var src = img.src; | ||||
| 					var coord = src.match(re); | ||||
| 					if (coord == null) { | ||||
| 						continue; | ||||
| 					} | ||||
| 					var y = parseInt(coord[1]); | ||||
| 					var x = parseInt(coord[2]); | ||||
| 					var z = parseInt(coord[3]); | ||||
| 					if (invalidate(x, y, z)) { | ||||
| 						var idx = src.lastIndexOf('#'); | ||||
| 						if (idx >= 0) { | ||||
| 							src = src.substring(0, idx); | ||||
| 						} | ||||
| 						img.src = src + '#' + Math.random(); | ||||
| 					} | ||||
| 				} | ||||
| 			}; | ||||
| 		} | ||||
| 	} | ||||
| }) | ||||
							
								
								
									
										4
									
								
								cmd/mtwebmapper/client/vue.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,4 @@ | ||||
| const { defineConfig } = require('@vue/cli-service') | ||||
| module.exports = defineConfig({ | ||||
|   transpileDependencies: true | ||||
| }) | ||||
							
								
								
									
										6159
									
								
								cmd/mtwebmapper/client/yarn.lock
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -1 +0,0 @@ | ||||
| .leaflet-control-coordinates{background-color:#D8D8D8;background-color:rgba(255,255,255,.8);cursor:pointer}.leaflet-control-coordinates,.leaflet-control-coordinates .uiElement input{-webkit-border-radius:5px;-moz-border-radius:5px;border-radius:5px}.leaflet-control-coordinates .uiElement{margin:4px}.leaflet-control-coordinates .uiElement .labelFirst{margin-right:4px}.leaflet-control-coordinates .uiHidden{display:none} | ||||
							
								
								
									
										1338
									
								
								cmd/mtwebmapper/web/css/font-awesome.css
									
									
									
									
										vendored
									
									
								
							
							
						
						| Before Width: | Height: | Size: 14 KiB | 
| Before Width: | Height: | Size: 30 KiB | 
| Before Width: | Height: | Size: 7.8 KiB | 
| Before Width: | Height: | Size: 535 B | 
| Before Width: | Height: | Size: 1.4 KiB | 
| Before Width: | Height: | Size: 40 KiB | 
| Before Width: | Height: | Size: 65 KiB | 
| @@ -1,124 +0,0 @@ | ||||
| /* | ||||
| Author: L. Voogdt | ||||
| License: MIT | ||||
| Version: 1.0 | ||||
| */ | ||||
|  | ||||
| /* Marker setup */ | ||||
| .awesome-marker { | ||||
|   background: url('images/markers-soft.png') no-repeat 0 0; | ||||
|   width: 35px; | ||||
|   height: 46px; | ||||
|   position:absolute; | ||||
|   left:0; | ||||
|   top:0; | ||||
|   display: block; | ||||
|   text-align: center; | ||||
| } | ||||
|  | ||||
| .awesome-marker-shadow { | ||||
|   background: url('images/markers-shadow.png') no-repeat 0 0; | ||||
|   width: 36px; | ||||
|   height: 16px; | ||||
| } | ||||
|  | ||||
| /* Retina displays */ | ||||
| @media (min--moz-device-pixel-ratio: 1.5),(-o-min-device-pixel-ratio: 3/2), | ||||
| (-webkit-min-device-pixel-ratio: 1.5),(min-device-pixel-ratio: 1.5),(min-resolution: 1.5dppx) { | ||||
|  .awesome-marker { | ||||
|   background-image: url('images/markers-soft@2x.png'); | ||||
|   background-size: 720px 46px; | ||||
|  } | ||||
|  .awesome-marker-shadow { | ||||
|   background-image: url('images/markers-shadow@2x.png'); | ||||
|   background-size: 35px 16px; | ||||
|  } | ||||
| } | ||||
|  | ||||
| .awesome-marker i { | ||||
|   color: #333; | ||||
|   margin-top: 10px; | ||||
|   display: inline-block; | ||||
|   font-size: 14px; | ||||
| } | ||||
|  | ||||
| .awesome-marker .icon-white { | ||||
|   color: #fff; | ||||
| } | ||||
|  | ||||
| /* Colors */ | ||||
| .awesome-marker-icon-red { | ||||
|   background-position: 0 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-darkred { | ||||
|   background-position: -180px 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-lightred { | ||||
|   background-position: -360px 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-orange { | ||||
|   background-position: -36px 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-beige { | ||||
|   background-position: -396px 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-green { | ||||
|   background-position: -72px 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-darkgreen { | ||||
|   background-position: -252px 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-lightgreen { | ||||
|   background-position: -432px 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-blue { | ||||
|   background-position: -108px 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-darkblue { | ||||
|   background-position: -216px 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-lightblue { | ||||
|   background-position: -468px 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-purple { | ||||
|   background-position: -144px 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-darkpurple { | ||||
|   background-position: -288px 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-pink { | ||||
|   background-position: -504px 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-cadetblue { | ||||
|   background-position: -324px 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-white { | ||||
|   background-position: -574px 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-gray { | ||||
|   background-position: -648px 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-lightgray { | ||||
|   background-position: -612px 0; | ||||
| } | ||||
|  | ||||
| .awesome-marker-icon-black { | ||||
|   background-position: -682px 0; | ||||
| } | ||||
| @@ -1,478 +0,0 @@ | ||||
| /* required styles */ | ||||
|  | ||||
| .leaflet-map-pane, | ||||
| .leaflet-tile, | ||||
| .leaflet-marker-icon, | ||||
| .leaflet-marker-shadow, | ||||
| .leaflet-tile-pane, | ||||
| .leaflet-tile-container, | ||||
| .leaflet-overlay-pane, | ||||
| .leaflet-shadow-pane, | ||||
| .leaflet-marker-pane, | ||||
| .leaflet-popup-pane, | ||||
| .leaflet-overlay-pane svg, | ||||
| .leaflet-zoom-box, | ||||
| .leaflet-image-layer, | ||||
| .leaflet-layer { | ||||
| 	position: absolute; | ||||
| 	left: 0; | ||||
| 	top: 0; | ||||
| 	} | ||||
| .leaflet-container { | ||||
| 	overflow: hidden; | ||||
| 	-ms-touch-action: none; | ||||
| 	} | ||||
| .leaflet-tile, | ||||
| .leaflet-marker-icon, | ||||
| .leaflet-marker-shadow { | ||||
| 	-webkit-user-select: none; | ||||
| 	   -moz-user-select: none; | ||||
| 	        user-select: none; | ||||
| 	-webkit-user-drag: none; | ||||
| 	} | ||||
| .leaflet-marker-icon, | ||||
| .leaflet-marker-shadow { | ||||
| 	display: block; | ||||
| 	} | ||||
| /* map is broken in FF if you have max-width: 100% on tiles */ | ||||
| .leaflet-container img { | ||||
| 	max-width: none !important; | ||||
| 	} | ||||
| /* stupid Android 2 doesn't understand "max-width: none" properly */ | ||||
| .leaflet-container img.leaflet-image-layer { | ||||
| 	max-width: 15000px !important; | ||||
| 	} | ||||
| .leaflet-tile { | ||||
| 	filter: inherit; | ||||
| 	visibility: hidden; | ||||
| 	} | ||||
| .leaflet-tile-loaded { | ||||
| 	visibility: inherit; | ||||
| 	} | ||||
| .leaflet-zoom-box { | ||||
| 	width: 0; | ||||
| 	height: 0; | ||||
| 	} | ||||
| /* workaround for https://bugzilla.mozilla.org/show_bug.cgi?id=888319 */ | ||||
| .leaflet-overlay-pane svg { | ||||
| 	-moz-user-select: none; | ||||
| 	} | ||||
|  | ||||
| .leaflet-tile-pane    { z-index: 2; } | ||||
| .leaflet-objects-pane { z-index: 3; } | ||||
| .leaflet-overlay-pane { z-index: 4; } | ||||
| .leaflet-shadow-pane  { z-index: 5; } | ||||
| .leaflet-marker-pane  { z-index: 6; } | ||||
| .leaflet-popup-pane   { z-index: 7; } | ||||
|  | ||||
| .leaflet-vml-shape { | ||||
| 	width: 1px; | ||||
| 	height: 1px; | ||||
| 	} | ||||
| .lvml { | ||||
| 	behavior: url(#default#VML); | ||||
| 	display: inline-block; | ||||
| 	position: absolute; | ||||
| 	} | ||||
|  | ||||
|  | ||||
| /* control positioning */ | ||||
|  | ||||
| .leaflet-control { | ||||
| 	position: relative; | ||||
| 	z-index: 7; | ||||
| 	pointer-events: auto; | ||||
| 	} | ||||
| .leaflet-top, | ||||
| .leaflet-bottom { | ||||
| 	position: absolute; | ||||
| 	z-index: 1000; | ||||
| 	pointer-events: none; | ||||
| 	} | ||||
| .leaflet-top { | ||||
| 	top: 0; | ||||
| 	} | ||||
| .leaflet-right { | ||||
| 	right: 0; | ||||
| 	} | ||||
| .leaflet-bottom { | ||||
| 	bottom: 0; | ||||
| 	} | ||||
| .leaflet-left { | ||||
| 	left: 0; | ||||
| 	} | ||||
| .leaflet-control { | ||||
| 	float: left; | ||||
| 	clear: both; | ||||
| 	} | ||||
| .leaflet-right .leaflet-control { | ||||
| 	float: right; | ||||
| 	} | ||||
| .leaflet-top .leaflet-control { | ||||
| 	margin-top: 10px; | ||||
| 	} | ||||
| .leaflet-bottom .leaflet-control { | ||||
| 	margin-bottom: 10px; | ||||
| 	} | ||||
| .leaflet-left .leaflet-control { | ||||
| 	margin-left: 10px; | ||||
| 	} | ||||
| .leaflet-right .leaflet-control { | ||||
| 	margin-right: 10px; | ||||
| 	} | ||||
|  | ||||
|  | ||||
| /* zoom and fade animations */ | ||||
|  | ||||
| .leaflet-fade-anim .leaflet-tile, | ||||
| .leaflet-fade-anim .leaflet-popup { | ||||
| 	opacity: 0; | ||||
| 	-webkit-transition: opacity 0.2s linear; | ||||
| 	   -moz-transition: opacity 0.2s linear; | ||||
| 	     -o-transition: opacity 0.2s linear; | ||||
| 	        transition: opacity 0.2s linear; | ||||
| 	} | ||||
| .leaflet-fade-anim .leaflet-tile-loaded, | ||||
| .leaflet-fade-anim .leaflet-map-pane .leaflet-popup { | ||||
| 	opacity: 1; | ||||
| 	} | ||||
|  | ||||
| .leaflet-zoom-anim .leaflet-zoom-animated { | ||||
| 	-webkit-transition: -webkit-transform 0.25s cubic-bezier(0,0,0.25,1); | ||||
| 	   -moz-transition:    -moz-transform 0.25s cubic-bezier(0,0,0.25,1); | ||||
| 	     -o-transition:      -o-transform 0.25s cubic-bezier(0,0,0.25,1); | ||||
| 	        transition:         transform 0.25s cubic-bezier(0,0,0.25,1); | ||||
| 	} | ||||
| .leaflet-zoom-anim .leaflet-tile, | ||||
| .leaflet-pan-anim .leaflet-tile, | ||||
| .leaflet-touching .leaflet-zoom-animated { | ||||
| 	-webkit-transition: none; | ||||
| 	   -moz-transition: none; | ||||
| 	     -o-transition: none; | ||||
| 	        transition: none; | ||||
| 	} | ||||
|  | ||||
| .leaflet-zoom-anim .leaflet-zoom-hide { | ||||
| 	visibility: hidden; | ||||
| 	} | ||||
|  | ||||
|  | ||||
| /* cursors */ | ||||
|  | ||||
| .leaflet-clickable { | ||||
| 	cursor: pointer; | ||||
| 	} | ||||
| .leaflet-container { | ||||
| 	cursor: -webkit-grab; | ||||
| 	cursor:    -moz-grab; | ||||
| 	} | ||||
| .leaflet-popup-pane, | ||||
| .leaflet-control { | ||||
| 	cursor: auto; | ||||
| 	} | ||||
| .leaflet-dragging .leaflet-container, | ||||
| .leaflet-dragging .leaflet-clickable { | ||||
| 	cursor: move; | ||||
| 	cursor: -webkit-grabbing; | ||||
| 	cursor:    -moz-grabbing; | ||||
| 	} | ||||
|  | ||||
|  | ||||
| /* visual tweaks */ | ||||
|  | ||||
| .leaflet-container { | ||||
| 	background: #ddd; | ||||
| 	outline: 0; | ||||
| 	} | ||||
| .leaflet-container a { | ||||
| 	color: #0078A8; | ||||
| 	} | ||||
| .leaflet-container a.leaflet-active { | ||||
| 	outline: 2px solid orange; | ||||
| 	} | ||||
| .leaflet-zoom-box { | ||||
| 	border: 2px dotted #38f; | ||||
| 	background: rgba(255,255,255,0.5); | ||||
| 	} | ||||
|  | ||||
|  | ||||
| /* general typography */ | ||||
| .leaflet-container { | ||||
| 	font: 12px/1.5 "Helvetica Neue", Arial, Helvetica, sans-serif; | ||||
| 	} | ||||
|  | ||||
|  | ||||
| /* general toolbar styles */ | ||||
|  | ||||
| .leaflet-bar { | ||||
| 	box-shadow: 0 1px 5px rgba(0,0,0,0.65); | ||||
| 	border-radius: 4px; | ||||
| 	} | ||||
| .leaflet-bar a, | ||||
| .leaflet-bar a:hover { | ||||
| 	background-color: #fff; | ||||
| 	border-bottom: 1px solid #ccc; | ||||
| 	width: 26px; | ||||
| 	height: 26px; | ||||
| 	line-height: 26px; | ||||
| 	display: block; | ||||
| 	text-align: center; | ||||
| 	text-decoration: none; | ||||
| 	color: black; | ||||
| 	} | ||||
| .leaflet-bar a, | ||||
| .leaflet-control-layers-toggle { | ||||
| 	background-position: 50% 50%; | ||||
| 	background-repeat: no-repeat; | ||||
| 	display: block; | ||||
| 	} | ||||
| .leaflet-bar a:hover { | ||||
| 	background-color: #f4f4f4; | ||||
| 	} | ||||
| .leaflet-bar a:first-child { | ||||
| 	border-top-left-radius: 4px; | ||||
| 	border-top-right-radius: 4px; | ||||
| 	} | ||||
| .leaflet-bar a:last-child { | ||||
| 	border-bottom-left-radius: 4px; | ||||
| 	border-bottom-right-radius: 4px; | ||||
| 	border-bottom: none; | ||||
| 	} | ||||
| .leaflet-bar a.leaflet-disabled { | ||||
| 	cursor: default; | ||||
| 	background-color: #f4f4f4; | ||||
| 	color: #bbb; | ||||
| 	} | ||||
|  | ||||
| .leaflet-touch .leaflet-bar a { | ||||
| 	width: 30px; | ||||
| 	height: 30px; | ||||
| 	line-height: 30px; | ||||
| 	} | ||||
|  | ||||
|  | ||||
| /* zoom control */ | ||||
|  | ||||
| .leaflet-control-zoom-in, | ||||
| .leaflet-control-zoom-out { | ||||
| 	font: bold 18px 'Lucida Console', Monaco, monospace; | ||||
| 	text-indent: 1px; | ||||
| 	} | ||||
| .leaflet-control-zoom-out { | ||||
| 	font-size: 20px; | ||||
| 	} | ||||
|  | ||||
| .leaflet-touch .leaflet-control-zoom-in { | ||||
| 	font-size: 22px; | ||||
| 	} | ||||
| .leaflet-touch .leaflet-control-zoom-out { | ||||
| 	font-size: 24px; | ||||
| 	} | ||||
|  | ||||
|  | ||||
| /* layers control */ | ||||
|  | ||||
| .leaflet-control-layers { | ||||
| 	box-shadow: 0 1px 5px rgba(0,0,0,0.4); | ||||
| 	background: #fff; | ||||
| 	border-radius: 5px; | ||||
| 	} | ||||
| .leaflet-control-layers-toggle { | ||||
| 	background-image: url(images/layers.png); | ||||
| 	width: 36px; | ||||
| 	height: 36px; | ||||
| 	} | ||||
| .leaflet-retina .leaflet-control-layers-toggle { | ||||
| 	background-image: url(images/layers-2x.png); | ||||
| 	background-size: 26px 26px; | ||||
| 	} | ||||
| .leaflet-touch .leaflet-control-layers-toggle { | ||||
| 	width: 44px; | ||||
| 	height: 44px; | ||||
| 	} | ||||
| .leaflet-control-layers .leaflet-control-layers-list, | ||||
| .leaflet-control-layers-expanded .leaflet-control-layers-toggle { | ||||
| 	display: none; | ||||
| 	} | ||||
| .leaflet-control-layers-expanded .leaflet-control-layers-list { | ||||
| 	display: block; | ||||
| 	position: relative; | ||||
| 	} | ||||
| .leaflet-control-layers-expanded { | ||||
| 	padding: 6px 10px 6px 6px; | ||||
| 	color: #333; | ||||
| 	background: #fff; | ||||
| 	} | ||||
| .leaflet-control-layers-selector { | ||||
| 	margin-top: 2px; | ||||
| 	position: relative; | ||||
| 	top: 1px; | ||||
| 	} | ||||
| .leaflet-control-layers label { | ||||
| 	display: block; | ||||
| 	} | ||||
| .leaflet-control-layers-separator { | ||||
| 	height: 0; | ||||
| 	border-top: 1px solid #ddd; | ||||
| 	margin: 5px -10px 5px -6px; | ||||
| 	} | ||||
|  | ||||
|  | ||||
| /* attribution and scale controls */ | ||||
|  | ||||
| .leaflet-container .leaflet-control-attribution { | ||||
| 	background: #fff; | ||||
| 	background: rgba(255, 255, 255, 0.7); | ||||
| 	margin: 0; | ||||
| 	} | ||||
| .leaflet-control-attribution, | ||||
| .leaflet-control-scale-line { | ||||
| 	padding: 0 5px; | ||||
| 	color: #333; | ||||
| 	} | ||||
| .leaflet-control-attribution a { | ||||
| 	text-decoration: none; | ||||
| 	} | ||||
| .leaflet-control-attribution a:hover { | ||||
| 	text-decoration: underline; | ||||
| 	} | ||||
| .leaflet-container .leaflet-control-attribution, | ||||
| .leaflet-container .leaflet-control-scale { | ||||
| 	font-size: 11px; | ||||
| 	} | ||||
| .leaflet-left .leaflet-control-scale { | ||||
| 	margin-left: 5px; | ||||
| 	} | ||||
| .leaflet-bottom .leaflet-control-scale { | ||||
| 	margin-bottom: 5px; | ||||
| 	} | ||||
| .leaflet-control-scale-line { | ||||
| 	border: 2px solid #777; | ||||
| 	border-top: none; | ||||
| 	line-height: 1.1; | ||||
| 	padding: 2px 5px 1px; | ||||
| 	font-size: 11px; | ||||
| 	white-space: nowrap; | ||||
| 	overflow: hidden; | ||||
| 	-moz-box-sizing: content-box; | ||||
| 	     box-sizing: content-box; | ||||
|  | ||||
| 	background: #fff; | ||||
| 	background: rgba(255, 255, 255, 0.5); | ||||
| 	} | ||||
| .leaflet-control-scale-line:not(:first-child) { | ||||
| 	border-top: 2px solid #777; | ||||
| 	border-bottom: none; | ||||
| 	margin-top: -2px; | ||||
| 	} | ||||
| .leaflet-control-scale-line:not(:first-child):not(:last-child) { | ||||
| 	border-bottom: 2px solid #777; | ||||
| 	} | ||||
|  | ||||
| .leaflet-touch .leaflet-control-attribution, | ||||
| .leaflet-touch .leaflet-control-layers, | ||||
| .leaflet-touch .leaflet-bar { | ||||
| 	box-shadow: none; | ||||
| 	} | ||||
| .leaflet-touch .leaflet-control-layers, | ||||
| .leaflet-touch .leaflet-bar { | ||||
| 	border: 2px solid rgba(0,0,0,0.2); | ||||
| 	background-clip: padding-box; | ||||
| 	} | ||||
|  | ||||
|  | ||||
| /* popup */ | ||||
|  | ||||
| .leaflet-popup { | ||||
| 	position: absolute; | ||||
| 	text-align: center; | ||||
| 	} | ||||
| .leaflet-popup-content-wrapper { | ||||
| 	padding: 1px; | ||||
| 	text-align: left; | ||||
| 	border-radius: 12px; | ||||
| 	} | ||||
| .leaflet-popup-content { | ||||
| 	margin: 13px 19px; | ||||
| 	line-height: 1.4; | ||||
| 	} | ||||
| .leaflet-popup-content p { | ||||
| 	margin: 18px 0; | ||||
| 	} | ||||
| .leaflet-popup-tip-container { | ||||
| 	margin: 0 auto; | ||||
| 	width: 40px; | ||||
| 	height: 20px; | ||||
| 	position: relative; | ||||
| 	overflow: hidden; | ||||
| 	} | ||||
| .leaflet-popup-tip { | ||||
| 	width: 17px; | ||||
| 	height: 17px; | ||||
| 	padding: 1px; | ||||
|  | ||||
| 	margin: -10px auto 0; | ||||
|  | ||||
| 	-webkit-transform: rotate(45deg); | ||||
| 	   -moz-transform: rotate(45deg); | ||||
| 	    -ms-transform: rotate(45deg); | ||||
| 	     -o-transform: rotate(45deg); | ||||
| 	        transform: rotate(45deg); | ||||
| 	} | ||||
| .leaflet-popup-content-wrapper, | ||||
| .leaflet-popup-tip { | ||||
| 	background: white; | ||||
|  | ||||
| 	box-shadow: 0 3px 14px rgba(0,0,0,0.4); | ||||
| 	} | ||||
| .leaflet-container a.leaflet-popup-close-button { | ||||
| 	position: absolute; | ||||
| 	top: 0; | ||||
| 	right: 0; | ||||
| 	padding: 4px 4px 0 0; | ||||
| 	text-align: center; | ||||
| 	width: 18px; | ||||
| 	height: 14px; | ||||
| 	font: 16px/14px Tahoma, Verdana, sans-serif; | ||||
| 	color: #c3c3c3; | ||||
| 	text-decoration: none; | ||||
| 	font-weight: bold; | ||||
| 	background: transparent; | ||||
| 	} | ||||
| .leaflet-container a.leaflet-popup-close-button:hover { | ||||
| 	color: #999; | ||||
| 	} | ||||
| .leaflet-popup-scrolled { | ||||
| 	overflow: auto; | ||||
| 	border-bottom: 1px solid #ddd; | ||||
| 	border-top: 1px solid #ddd; | ||||
| 	} | ||||
|  | ||||
| .leaflet-oldie .leaflet-popup-content-wrapper { | ||||
| 	zoom: 1; | ||||
| 	} | ||||
| .leaflet-oldie .leaflet-popup-tip { | ||||
| 	width: 24px; | ||||
| 	margin: 0 auto; | ||||
|  | ||||
| 	-ms-filter: "progid:DXImageTransform.Microsoft.Matrix(M11=0.70710678, M12=0.70710678, M21=-0.70710678, M22=0.70710678)"; | ||||
| 	filter: progid:DXImageTransform.Microsoft.Matrix(M11=0.70710678, M12=0.70710678, M21=-0.70710678, M22=0.70710678); | ||||
| 	} | ||||
| .leaflet-oldie .leaflet-popup-tip-container { | ||||
| 	margin-top: -1px; | ||||
| 	} | ||||
|  | ||||
| .leaflet-oldie .leaflet-control-zoom, | ||||
| .leaflet-oldie .leaflet-control-layers, | ||||
| .leaflet-oldie .leaflet-popup-content-wrapper, | ||||
| .leaflet-oldie .leaflet-popup-tip { | ||||
| 	border: 1px solid #999; | ||||
| 	} | ||||
|  | ||||
|  | ||||
| /* div icon */ | ||||
|  | ||||
| .leaflet-div-icon { | ||||
| 	background: #fff; | ||||
| 	border: 1px solid #666; | ||||
| 	} | ||||
| @@ -1,152 +0,0 @@ | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
| <head> | ||||
| <title>Minetest demo map</title> | ||||
| <meta charset="utf-8" /> | ||||
|     <link rel="stylesheet" href="css/leaflet.css" /> | ||||
|     <link rel="stylesheet" href="css/Leaflet.Coordinates-0.1.4.css" /> | ||||
|     <link rel="stylesheet" href="css/font-awesome.css" /> | ||||
|     <link rel="stylesheet" href="css/leaflet.awesome-markers.css" /> | ||||
|     <style type="text/css"> | ||||
|         body { | ||||
|             height: 100%; | ||||
|         } | ||||
|  | ||||
|         #map { | ||||
|             display: block; | ||||
|             position: absolute; | ||||
|             top: 0; | ||||
|             left: 0; | ||||
|             width: 100%; | ||||
|             height: 100%; | ||||
|             background-color: #111111; | ||||
|         } | ||||
|  | ||||
|         .leaflet-container { | ||||
|             cursor: crosshair; | ||||
|         } | ||||
|  | ||||
|         .leaflet-control-coordinates, | ||||
|         .leaflet-control-layers { | ||||
|             box-shadow: 0 1px 3px rgba(0,0,0,0.3); | ||||
|             background-color:rgba(255,255,255,.85); | ||||
|         } | ||||
|         .awesome-marker i { | ||||
|             font-size: 18px; | ||||
|             margin-left: -1px; | ||||
|         } | ||||
|     </style> | ||||
| </head> | ||||
| <body> | ||||
|     <div id="map"></div> | ||||
|     <script src="js/leaflet.js"></script> | ||||
|     <script src="js/Leaflet.Coordinates-0.1.4.min.js"></script> | ||||
|     <script src="js/easy-button.js"></script> | ||||
|     <script src="js/auto-update.js"></script> | ||||
|     <script type="text/javascript" src="js/leaflet-hash.js"></script> | ||||
|     <script type="text/javascript" src="js/leaflet.ajax.js"></script> | ||||
|     <script type="text/javascript" src="js/leaflet.awesome-markers.js"></script> | ||||
|     <script> | ||||
|  | ||||
| var useWebsocket = true; // Set to true if you want websocket support | ||||
|  | ||||
| L.Projection.NoWrap = { | ||||
|     project: function (latlng) { | ||||
|         return new L.Point(latlng.lat, latlng.lng); | ||||
|     }, | ||||
|  | ||||
|     unproject: function (point, unbounded) { | ||||
|         return new L.LatLng(point.x, point.y, true); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| L.CRS.Direct = L.Util.extend({}, L.CRS, { | ||||
|     code: 'Direct', | ||||
|     projection: L.Projection.NoWrap, | ||||
|     transformation: new L.Transformation(1.0/65536, 30928.0/65536, -1.0/65536, 34608.0/65536) | ||||
| }); | ||||
|  | ||||
| var world = new L.tileLayer('map/{z}/{x}/{y}.png', { | ||||
|     minZoom: 0, | ||||
|     maxZoom: 16, | ||||
|     attribution: 'Demo world', | ||||
|     continuousWorld: false, | ||||
|     noWrap: true, | ||||
|     tms: true, | ||||
|     unloadInvisibleTiles: true | ||||
| }); | ||||
|  | ||||
| var players = L.geoJson.ajax('/players', { | ||||
|     pointToLayer: function(feature, latlng) { | ||||
|         return L.marker(latlng, { | ||||
|             icon: L.AwesomeMarkers.icon({ | ||||
|                 icon: 'male', | ||||
|                 iconColor: 'black', | ||||
|                 prefix: 'fa', | ||||
|                 markerColor: 'orange' | ||||
|             }), | ||||
|             title: feature.properties.name | ||||
|         }) | ||||
|     } | ||||
| }); | ||||
| var rasterMaps = { | ||||
|     "A demo world": world, | ||||
| }; | ||||
|  | ||||
| var latest = world | ||||
|  | ||||
| var overlayMaps = {'Players': players}; | ||||
|  | ||||
| var map = L.map('map', { | ||||
|     center: [0,0], | ||||
|     zoom: 3, | ||||
|     layers: [latest], | ||||
|     worldCopyJump: false, | ||||
|     crs: L.CRS.Direct}); | ||||
|  | ||||
| L.control.coordinates({ | ||||
|     position:"topright", //optional default "bootomright" | ||||
|     decimals:0, //optional default 4 | ||||
|     decimalSeperator:".", //optional default "." | ||||
|     labelTemplateLat:"X: {y}", //optional default "Lat: {y}" | ||||
|     labelTemplateLng:"Y: {x}", //optional default "Lng: {x}" | ||||
|     enableUserInput:false, //optional default true | ||||
|     useDMS:false, //optional default false | ||||
|     useLatLngOrder: true //ordering of labels, default false-> lng-lat | ||||
| }).addTo(map); | ||||
|  | ||||
| var manualUpdateControl; | ||||
| if (useWebsocket && 'WebSocket' in window) { | ||||
|     L.autoUpdate('autoUpdate', function(pressed) { | ||||
|         var styleDec = manualUpdateControl.getContainer().style; | ||||
|         styleDec.visibility = pressed ? 'hidden' : 'visible'; | ||||
|     }, | ||||
|     players); | ||||
| } | ||||
| var layersControl = new L.Control.Layers(rasterMaps, overlayMaps, {collapsed: false}); | ||||
| map.addControl(layersControl); | ||||
|  | ||||
| manualUpdateControl = L.easyButton('fa-refresh', | ||||
|     function (){ | ||||
|         var tiles = document.getElementsByTagName("img"); | ||||
|         for (var i = 0; i < tiles.length; i++) { | ||||
|             var img = tiles[i]; | ||||
|             var cl = img.getAttribute("class"); | ||||
|             if (cl.indexOf("leaflet-tile-loaded") >= 0) { | ||||
|                 var src = img.src; | ||||
|                 var idx = src.lastIndexOf("#"); | ||||
|                 if (idx >= 0) { | ||||
|                     src = src.substring(0, idx); | ||||
|                 } | ||||
|                 img.src = src + "#" + Math.random(); | ||||
|             } | ||||
|         } | ||||
|         //map._resetView(map.getCenter(), map.getZoom(), false); | ||||
|         players.refresh("/players"); | ||||
|     }, | ||||
|     'Update view' | ||||
| ); | ||||
|     var hash = new L.Hash(map) | ||||
|     </script> | ||||
| </body> | ||||
| </html> | ||||
| @@ -1,187 +0,0 @@ | ||||
| L.Control.AutoUpdate = L.Control.extend({ | ||||
|     options: { | ||||
|         position: 'topleft', | ||||
|         label: 'Automatic update', | ||||
|         layer: undefined | ||||
|     }, | ||||
|     pressed: true, | ||||
|  | ||||
|     onAdd: function() { | ||||
|         var container = L.DomUtil.create('div', 'leaflet-bar leaflet-control'); | ||||
|         this.link = L.DomUtil.create('a', 'leaflet-bar-part', container); | ||||
|         this.iconStart = L.DomUtil.create('i', 'fa fa-play', this.link); | ||||
|         this.link.href = '#'; | ||||
|         L.DomEvent.on(this.link, 'click', this.cbClick, this); | ||||
|         return container; | ||||
|     }, | ||||
|  | ||||
|     switchButtons: function() { | ||||
|         if (this.pressed) { | ||||
|             this.pressed = false; | ||||
|             this.iconStart.setAttribute('class', 'fa fa-pause'); | ||||
|             this.autoUpdate(); | ||||
|         } else { | ||||
|             this.pressed = true; | ||||
|             this.iconStart.setAttribute('class', 'fa fa-play'); | ||||
|             this.stopUpdate(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     cbClick: function (e) { | ||||
|         L.DomEvent.stopPropagation(e); | ||||
|         this.intendedFunction(this.pressed); | ||||
|         this.switchButtons(); | ||||
|     }, | ||||
|  | ||||
|     intendedFunction: function() { | ||||
|         alert('no function selected'); | ||||
|     }, | ||||
|  | ||||
|     stopUpdate: function() { | ||||
|         if (this.socket) { | ||||
|             var s = this.socket; | ||||
|             this.socket = null; | ||||
|             s.close(); | ||||
|         } | ||||
|     }, | ||||
|  | ||||
|     autoUpdate: function() { | ||||
|         var me = this; | ||||
|         this.socket = new WebSocket('ws://' + window.location.host + '/ws'); | ||||
|  | ||||
|         this.socket.onerror = function(evt) { | ||||
|             me.stopUpdate(); | ||||
|             me.switchButtons(); | ||||
|         }; | ||||
|  | ||||
|         this.socket.onclose = function(evt) { | ||||
|             this.socket = null; | ||||
|         } | ||||
|  | ||||
|         this.socket.onopen = function(evt) { | ||||
|             // Sending pings every 5 secs to keep connection alive. | ||||
|             var heartbeat = function() { | ||||
|                 if (heartbeat && me.socket) { | ||||
|                     me.socket.send("PING"); | ||||
|                     setTimeout(heartbeat, 8000); | ||||
|                 } else { | ||||
|                     // Prevent sending pings to re-opened sockets. | ||||
|                     heartbeat = null; | ||||
|                 } | ||||
|             }; | ||||
|             setTimeout(heartbeat, 8000); | ||||
|         }; | ||||
|  | ||||
|         this.socket.onmessage = function(evt) { | ||||
|  | ||||
|             var json = evt.data; | ||||
|             if (!(typeof json === "string")) { | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             var msg; | ||||
|             try { | ||||
|                 msg = JSON.parse(json); | ||||
|             } | ||||
|             catch (err) { | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             if (msg.players) { | ||||
|                 me.options.layer.clearLayers(); | ||||
|                 me.options.layer.addData(msg.players); | ||||
|             } | ||||
|  | ||||
|             var tilesData = msg.tiles; | ||||
|             if (!tilesData) { | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             var invalidate = function(td) { | ||||
|  | ||||
|                 var pyramid = new Array(9); | ||||
|                 var last = new Object(); | ||||
|                 pyramid[8] = last; | ||||
|  | ||||
|                 for (var i = 0; i < td.length; i++) { | ||||
|                     var xz = td[i]; | ||||
|                     last[xz.X + "#" + xz.Z] = xz; | ||||
|                 } | ||||
|                 for (var p = 7; p >= 0; p--) { | ||||
|                     var prev = pyramid[p+1]; | ||||
|                     var curr = new Object(); | ||||
|                     pyramid[p] = curr; | ||||
|                     for (var k in prev) { | ||||
|                         if (prev.hasOwnProperty(k)) { | ||||
|                             var oxz = prev[k]; | ||||
|                             var nxz = { X: oxz.X >> 1, Z: oxz.Z >> 1 }; | ||||
|                             curr[nxz.X + "#" + nxz.Z] = nxz; | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|                 return function(x, y, z) { | ||||
|                     if (y > 8) { | ||||
|                         x >>= y - 8; | ||||
|                         z >>= y - 8; | ||||
|                         y = 8; | ||||
|                     } | ||||
|                     var level = pyramid[y]; | ||||
|                     var k = x + "#" + z; | ||||
|                     return level.hasOwnProperty(k); | ||||
|                 }; | ||||
|             } (tilesData); | ||||
|  | ||||
|  | ||||
|             var tiles = document.getElementsByTagName('img'); | ||||
|             var re = /\/map\/([0-9]+)\/([0-9]+)\/([0-9]+).*/; | ||||
|             for (var i = 0; i < tiles.length; i++) { | ||||
|                 var img = tiles[i]; | ||||
|                 var cl = img.getAttribute('class'); | ||||
|                 if (cl.indexOf('leaflet-tile-loaded') < 0) { | ||||
|                     continue; | ||||
|                 } | ||||
|                 var src = img.src; | ||||
|                 var coord = src.match(re); | ||||
|                 if (coord == null) { | ||||
|                     continue; | ||||
|                 } | ||||
|                 var y = parseInt(coord[1]); | ||||
|                 var x = parseInt(coord[2]); | ||||
|                 var z = parseInt(coord[3]); | ||||
|                 if (invalidate(x, y, z)) { | ||||
|                     var idx = src.lastIndexOf('#'); | ||||
|                     if (idx >= 0) { | ||||
|                         src = src.substring(0, idx); | ||||
|                     } | ||||
|                     img.src = src + '#' + Math.random(); | ||||
|                 } | ||||
|             } | ||||
|         }; | ||||
|     } | ||||
| }); | ||||
|  | ||||
| L.autoUpdate = function(cbLabel, cbFunc, layer, cbMap) { | ||||
|     var control = new L.Control.AutoUpdate(); | ||||
|     if (cbLabel) { | ||||
|         control.options.label = cbLabel; | ||||
|     } | ||||
|  | ||||
|     if (cbFunc) { | ||||
|         control.intendedFunction = cbFunc; | ||||
|     } | ||||
|  | ||||
|     if (layer) { | ||||
|         control.options.layer = layer; | ||||
|     } | ||||
|  | ||||
|     if (cbMap === '') { | ||||
|         return control; | ||||
|     } | ||||
|     else if (cbMap) { | ||||
|         cbMap.addControl(control); | ||||
|     } | ||||
|     else { | ||||
|         map.addControl(control); | ||||
|     } | ||||
|     return control; | ||||
| }; | ||||
| @@ -1,48 +0,0 @@ | ||||
| L.Control.EasyButtons = L.Control.extend({ | ||||
|     options: { | ||||
|         position: 'topleft', | ||||
|         title: '', | ||||
|         intentedIcon: 'fa-circle-o' | ||||
|     }, | ||||
|  | ||||
|     onAdd: function () { | ||||
|         var container = L.DomUtil.create('div', 'leaflet-bar leaflet-control'); | ||||
|  | ||||
|         this.link = L.DomUtil.create('a', 'leaflet-bar-part', container); | ||||
|         L.DomUtil.create('i', 'fa fa-lg ' + this.options.intentedIcon , this.link); | ||||
|         this.link.href = '#'; | ||||
|  | ||||
|         L.DomEvent.on(this.link, 'click', this._click, this); | ||||
|         this.link.title = this.options.title; | ||||
|  | ||||
|         return container; | ||||
|     }, | ||||
|  | ||||
|     intendedFunction: function(){ alert('no function selected');}, | ||||
|  | ||||
|     _click: function (e) { | ||||
|         L.DomEvent.stopPropagation(e); | ||||
|         L.DomEvent.preventDefault(e); | ||||
|         this.intendedFunction(); | ||||
|     }, | ||||
| }); | ||||
|  | ||||
| L.easyButton = function( btnIcon , btnFunction , btnTitle , btnMap ) { | ||||
|   var newControl = new L.Control.EasyButtons; | ||||
|   if (btnIcon) newControl.options.intentedIcon = btnIcon; | ||||
|  | ||||
|   if ( typeof btnFunction === 'function'){ | ||||
|     newControl.intendedFunction = btnFunction; | ||||
|   } | ||||
|  | ||||
|   if (btnTitle) newControl.options.title = btnTitle; | ||||
|  | ||||
|   if ( btnMap == '' ){ | ||||
|     // skip auto addition | ||||
|   } else if ( btnMap ) { | ||||
|     btnMap.addControl(newControl); | ||||
|   } else { | ||||
|     map.addControl(newControl); | ||||
|   } | ||||
|   return newControl; | ||||
| }; | ||||
| @@ -1,162 +0,0 @@ | ||||
| (function(window) { | ||||
| 	var HAS_HASHCHANGE = (function() { | ||||
| 		var doc_mode = window.documentMode; | ||||
| 		return ('onhashchange' in window) && | ||||
| 			(doc_mode === undefined || doc_mode > 7); | ||||
| 	})(); | ||||
|  | ||||
| 	L.Hash = function(map) { | ||||
| 		this.onHashChange = L.Util.bind(this.onHashChange, this); | ||||
|  | ||||
| 		if (map) { | ||||
| 			this.init(map); | ||||
| 		} | ||||
| 	}; | ||||
|  | ||||
| 	L.Hash.parseHash = function(hash) { | ||||
| 		if(hash.indexOf('#') === 0) { | ||||
| 			hash = hash.substr(1); | ||||
| 		} | ||||
| 		var args = hash.split("/"); | ||||
| 		if (args.length == 3) { | ||||
| 			var zoom = parseInt(args[0], 10), | ||||
| 			lat = parseFloat(args[1]), | ||||
| 			lon = parseFloat(args[2]); | ||||
| 			if (isNaN(zoom) || isNaN(lat) || isNaN(lon)) { | ||||
| 				return false; | ||||
| 			} else { | ||||
| 				return { | ||||
| 					center: new L.LatLng(lat, lon), | ||||
| 					zoom: zoom | ||||
| 				}; | ||||
| 			} | ||||
| 		} else { | ||||
| 			return false; | ||||
| 		} | ||||
| 	}; | ||||
|  | ||||
| 	L.Hash.formatHash = function(map) { | ||||
| 		var center = map.getCenter(), | ||||
| 		    zoom = map.getZoom(), | ||||
| 		    precision = Math.max(0, Math.ceil(Math.log(zoom) / Math.LN2)); | ||||
|  | ||||
| 		return "#" + [zoom, | ||||
| 			center.lat.toFixed(precision), | ||||
| 			center.lng.toFixed(precision) | ||||
| 		].join("/"); | ||||
| 	}, | ||||
|  | ||||
| 	L.Hash.prototype = { | ||||
| 		map: null, | ||||
| 		lastHash: null, | ||||
|  | ||||
| 		parseHash: L.Hash.parseHash, | ||||
| 		formatHash: L.Hash.formatHash, | ||||
|  | ||||
| 		init: function(map) { | ||||
| 			this.map = map; | ||||
|  | ||||
| 			// reset the hash | ||||
| 			this.lastHash = null; | ||||
| 			this.onHashChange(); | ||||
|  | ||||
| 			if (!this.isListening) { | ||||
| 				this.startListening(); | ||||
| 			} | ||||
| 		}, | ||||
|  | ||||
| 		removeFrom: function(map) { | ||||
| 			if (this.changeTimeout) { | ||||
| 				clearTimeout(this.changeTimeout); | ||||
| 			} | ||||
|  | ||||
| 			if (this.isListening) { | ||||
| 				this.stopListening(); | ||||
| 			} | ||||
|  | ||||
| 			this.map = null; | ||||
| 		}, | ||||
|  | ||||
| 		onMapMove: function() { | ||||
| 			// bail if we're moving the map (updating from a hash), | ||||
| 			// or if the map is not yet loaded | ||||
|  | ||||
| 			if (this.movingMap || !this.map._loaded) { | ||||
| 				return false; | ||||
| 			} | ||||
|  | ||||
| 			var hash = this.formatHash(this.map); | ||||
| 			if (this.lastHash != hash) { | ||||
| 				location.replace(hash); | ||||
| 				this.lastHash = hash; | ||||
| 			} | ||||
| 		}, | ||||
|  | ||||
| 		movingMap: false, | ||||
| 		update: function() { | ||||
| 			var hash = location.hash; | ||||
| 			if (hash === this.lastHash) { | ||||
| 				return; | ||||
| 			} | ||||
| 			var parsed = this.parseHash(hash); | ||||
| 			if (parsed) { | ||||
| 				this.movingMap = true; | ||||
|  | ||||
| 				this.map.setView(parsed.center, parsed.zoom); | ||||
|  | ||||
| 				this.movingMap = false; | ||||
| 			} else { | ||||
| 				this.onMapMove(this.map); | ||||
| 			} | ||||
| 		}, | ||||
|  | ||||
| 		// defer hash change updates every 100ms | ||||
| 		changeDefer: 100, | ||||
| 		changeTimeout: null, | ||||
| 		onHashChange: function() { | ||||
| 			// throttle calls to update() so that they only happen every | ||||
| 			// `changeDefer` ms | ||||
| 			if (!this.changeTimeout) { | ||||
| 				var that = this; | ||||
| 				this.changeTimeout = setTimeout(function() { | ||||
| 					that.update(); | ||||
| 					that.changeTimeout = null; | ||||
| 				}, this.changeDefer); | ||||
| 			} | ||||
| 		}, | ||||
|  | ||||
| 		isListening: false, | ||||
| 		hashChangeInterval: null, | ||||
| 		startListening: function() { | ||||
| 			this.map.on("moveend", this.onMapMove, this); | ||||
|  | ||||
| 			if (HAS_HASHCHANGE) { | ||||
| 				L.DomEvent.addListener(window, "hashchange", this.onHashChange); | ||||
| 			} else { | ||||
| 				clearInterval(this.hashChangeInterval); | ||||
| 				this.hashChangeInterval = setInterval(this.onHashChange, 50); | ||||
| 			} | ||||
| 			this.isListening = true; | ||||
| 		}, | ||||
|  | ||||
| 		stopListening: function() { | ||||
| 			this.map.off("moveend", this.onMapMove, this); | ||||
|  | ||||
| 			if (HAS_HASHCHANGE) { | ||||
| 				L.DomEvent.removeListener(window, "hashchange", this.onHashChange); | ||||
| 			} else { | ||||
| 				clearInterval(this.hashChangeInterval); | ||||
| 			} | ||||
| 			this.isListening = false; | ||||
| 		} | ||||
| 	}; | ||||
| 	L.hash = function(map) { | ||||
| 		return new L.Hash(map); | ||||
| 	}; | ||||
| 	L.Map.prototype.addHash = function() { | ||||
| 		this._hash = L.hash(this); | ||||
| 	}; | ||||
| 	L.Map.prototype.removeHash = function() { | ||||
| 		this._hash.removeFrom(); | ||||
| 	}; | ||||
| })(window); | ||||
| @@ -1,740 +0,0 @@ | ||||
| ;(function(){ | ||||
|  | ||||
| /** | ||||
|  * Require the given path. | ||||
|  * | ||||
|  * @param {String} path | ||||
|  * @return {Object} exports | ||||
|  * @api public | ||||
|  */ | ||||
|  | ||||
| function require(path, parent, orig) { | ||||
|   var resolved = require.resolve(path); | ||||
|  | ||||
|   // lookup failed | ||||
|   if (null == resolved) { | ||||
|     orig = orig || path; | ||||
|     parent = parent || 'root'; | ||||
|     var err = new Error('Failed to require "' + orig + '" from "' + parent + '"'); | ||||
|     err.path = orig; | ||||
|     err.parent = parent; | ||||
|     err.require = true; | ||||
|     throw err; | ||||
|   } | ||||
|  | ||||
|   var module = require.modules[resolved]; | ||||
|  | ||||
|   // perform real require() | ||||
|   // by invoking the module's | ||||
|   // registered function | ||||
|   if (!module.exports) { | ||||
|     module.exports = {}; | ||||
|     module.client = module.component = true; | ||||
|     module.call(this, module.exports, require.relative(resolved), module); | ||||
|   } | ||||
|  | ||||
|   return module.exports; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Registered modules. | ||||
|  */ | ||||
|  | ||||
| require.modules = {}; | ||||
|  | ||||
| /** | ||||
|  * Registered aliases. | ||||
|  */ | ||||
|  | ||||
| require.aliases = {}; | ||||
|  | ||||
| /** | ||||
|  * Resolve `path`. | ||||
|  * | ||||
|  * Lookup: | ||||
|  * | ||||
|  *   - PATH/index.js | ||||
|  *   - PATH.js | ||||
|  *   - PATH | ||||
|  * | ||||
|  * @param {String} path | ||||
|  * @return {String} path or null | ||||
|  * @api private | ||||
|  */ | ||||
|  | ||||
| require.resolve = function(path) { | ||||
|   if (path.charAt(0) === '/') path = path.slice(1); | ||||
|  | ||||
|   var paths = [ | ||||
|     path, | ||||
|     path + '.js', | ||||
|     path + '.json', | ||||
|     path + '/index.js', | ||||
|     path + '/index.json' | ||||
|   ]; | ||||
|  | ||||
|   for (var i = 0; i < paths.length; i++) { | ||||
|     var path = paths[i]; | ||||
|     if (require.modules.hasOwnProperty(path)) return path; | ||||
|     if (require.aliases.hasOwnProperty(path)) return require.aliases[path]; | ||||
|   } | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Normalize `path` relative to the current path. | ||||
|  * | ||||
|  * @param {String} curr | ||||
|  * @param {String} path | ||||
|  * @return {String} | ||||
|  * @api private | ||||
|  */ | ||||
|  | ||||
| require.normalize = function(curr, path) { | ||||
|   var segs = []; | ||||
|  | ||||
|   if ('.' != path.charAt(0)) return path; | ||||
|  | ||||
|   curr = curr.split('/'); | ||||
|   path = path.split('/'); | ||||
|  | ||||
|   for (var i = 0; i < path.length; ++i) { | ||||
|     if ('..' == path[i]) { | ||||
|       curr.pop(); | ||||
|     } else if ('.' != path[i] && '' != path[i]) { | ||||
|       segs.push(path[i]); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   return curr.concat(segs).join('/'); | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Register module at `path` with callback `definition`. | ||||
|  * | ||||
|  * @param {String} path | ||||
|  * @param {Function} definition | ||||
|  * @api private | ||||
|  */ | ||||
|  | ||||
| require.register = function(path, definition) { | ||||
|   require.modules[path] = definition; | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Alias a module definition. | ||||
|  * | ||||
|  * @param {String} from | ||||
|  * @param {String} to | ||||
|  * @api private | ||||
|  */ | ||||
|  | ||||
| require.alias = function(from, to) { | ||||
|   if (!require.modules.hasOwnProperty(from)) { | ||||
|     throw new Error('Failed to alias "' + from + '", it does not exist'); | ||||
|   } | ||||
|   require.aliases[to] = from; | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Return a require function relative to the `parent` path. | ||||
|  * | ||||
|  * @param {String} parent | ||||
|  * @return {Function} | ||||
|  * @api private | ||||
|  */ | ||||
|  | ||||
| require.relative = function(parent) { | ||||
|   var p = require.normalize(parent, '..'); | ||||
|  | ||||
|   /** | ||||
|    * lastIndexOf helper. | ||||
|    */ | ||||
|  | ||||
|   function lastIndexOf(arr, obj) { | ||||
|     var i = arr.length; | ||||
|     while (i--) { | ||||
|       if (arr[i] === obj) return i; | ||||
|     } | ||||
|     return -1; | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * The relative require() itself. | ||||
|    */ | ||||
|  | ||||
|   function localRequire(path) { | ||||
|     var resolved = localRequire.resolve(path); | ||||
|     return require(resolved, parent, path); | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * Resolve relative to the parent. | ||||
|    */ | ||||
|  | ||||
|   localRequire.resolve = function(path) { | ||||
|     var c = path.charAt(0); | ||||
|     if ('/' == c) return path.slice(1); | ||||
|     if ('.' == c) return require.normalize(p, path); | ||||
|  | ||||
|     // resolve deps by returning | ||||
|     // the dep in the nearest "deps" | ||||
|     // directory | ||||
|     var segs = parent.split('/'); | ||||
|     var i = lastIndexOf(segs, 'deps') + 1; | ||||
|     if (!i) i = 0; | ||||
|     path = segs.slice(0, i + 1).join('/') + '/deps/' + path; | ||||
|     return path; | ||||
|   }; | ||||
|  | ||||
|   /** | ||||
|    * Check if module is defined at `path`. | ||||
|    */ | ||||
|  | ||||
|   localRequire.exists = function(path) { | ||||
|     return require.modules.hasOwnProperty(localRequire.resolve(path)); | ||||
|   }; | ||||
|  | ||||
|   return localRequire; | ||||
| }; | ||||
| require.register("calvinmetcalf-setImmediate/lib/index.js", function(exports, require, module){ | ||||
| "use strict"; | ||||
| var types = [ | ||||
|     require("./nextTick"), | ||||
|     require("./mutation"), | ||||
|     require("./postMessage"), | ||||
|     require("./messageChannel"), | ||||
|     require("./stateChange"), | ||||
|     require("./timeout") | ||||
| ]; | ||||
| var handlerQueue = []; | ||||
|  | ||||
| function drainQueue() { | ||||
|     var i = 0, | ||||
|         task, | ||||
|         innerQueue = handlerQueue; | ||||
| 	handlerQueue = []; | ||||
| 	/*jslint boss: true */ | ||||
| 	while (task = innerQueue[i++]) { | ||||
| 		task(); | ||||
| 	} | ||||
| } | ||||
| var nextTick; | ||||
| types.some(function (obj) { | ||||
|     var t = obj.test(); | ||||
|     if (t) { | ||||
|         nextTick = obj.install(drainQueue); | ||||
|     } | ||||
|     return t; | ||||
| }); | ||||
| var retFunc = function (task) { | ||||
|     var len, args; | ||||
|     if (arguments.length > 1 && typeof task === "function") { | ||||
|         args = Array.prototype.slice.call(arguments, 1); | ||||
|         args.unshift(undefined); | ||||
|         task = task.bind.apply(task, args); | ||||
|     } | ||||
|     if ((len = handlerQueue.push(task)) === 1) { | ||||
|         nextTick(drainQueue); | ||||
|     } | ||||
|     return len; | ||||
| }; | ||||
| retFunc.clear = function (n) { | ||||
|     if (n <= handlerQueue.length) { | ||||
|         handlerQueue[n - 1] = function () {}; | ||||
|     } | ||||
|     return this; | ||||
| }; | ||||
| module.exports = retFunc; | ||||
|  | ||||
| }); | ||||
| require.register("calvinmetcalf-setImmediate/lib/nextTick.js", function(exports, require, module){ | ||||
| "use strict"; | ||||
| exports.test = function () { | ||||
|     // Don't get fooled by e.g. browserify environments. | ||||
|     return typeof process === "object" && Object.prototype.toString.call(process) === "[object process]"; | ||||
| }; | ||||
|  | ||||
| exports.install = function () { | ||||
|     return process.nextTick; | ||||
| }; | ||||
| }); | ||||
| require.register("calvinmetcalf-setImmediate/lib/postMessage.js", function(exports, require, module){ | ||||
| "use strict"; | ||||
| var globe = require("./global"); | ||||
| exports.test = function () { | ||||
|     // The test against `importScripts` prevents this implementation from being installed inside a web worker, | ||||
|     // where `global.postMessage` means something completely different and can"t be used for this purpose. | ||||
|  | ||||
|     if (!globe.postMessage || globe.importScripts) { | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     var postMessageIsAsynchronous = true; | ||||
|     var oldOnMessage = globe.onmessage; | ||||
|     globe.onmessage = function () { | ||||
|         postMessageIsAsynchronous = false; | ||||
|     }; | ||||
|     globe.postMessage("", "*"); | ||||
|     globe.onmessage = oldOnMessage; | ||||
|  | ||||
|     return postMessageIsAsynchronous; | ||||
| }; | ||||
|  | ||||
| exports.install = function (func) { | ||||
|     var codeWord = "com.calvinmetcalf.setImmediate" + Math.random(); | ||||
|     function globalMessage(event) { | ||||
|         if (event.source === globe && event.data === codeWord) { | ||||
|             func(); | ||||
|         } | ||||
|     } | ||||
|     if (globe.addEventListener) { | ||||
|         globe.addEventListener("message", globalMessage, false); | ||||
|     } else { | ||||
|         globe.attachEvent("onmessage", globalMessage); | ||||
|     } | ||||
|     return function () { | ||||
|         globe.postMessage(codeWord, "*"); | ||||
|     }; | ||||
| }; | ||||
| }); | ||||
| require.register("calvinmetcalf-setImmediate/lib/messageChannel.js", function(exports, require, module){ | ||||
| "use strict"; | ||||
| var globe = require("./global"); | ||||
| exports.test = function () { | ||||
|     return !!globe.MessageChannel; | ||||
| }; | ||||
|  | ||||
| exports.install = function (func) { | ||||
|     var channel = new globe.MessageChannel(); | ||||
|     channel.port1.onmessage = func; | ||||
|     return function () { | ||||
|         channel.port2.postMessage(0); | ||||
|     }; | ||||
| }; | ||||
| }); | ||||
| require.register("calvinmetcalf-setImmediate/lib/stateChange.js", function(exports, require, module){ | ||||
| "use strict"; | ||||
| var globe = require("./global"); | ||||
| exports.test = function () { | ||||
|     return "document" in globe && "onreadystatechange" in globe.document.createElement("script"); | ||||
| }; | ||||
|  | ||||
| exports.install = function (handle) { | ||||
|     return function () { | ||||
|  | ||||
|         // Create a <script> element; its readystatechange event will be fired asynchronously once it is inserted | ||||
|         // into the document. Do so, thus queuing up the task. Remember to clean up once it's been called. | ||||
|         var scriptEl = globe.document.createElement("script"); | ||||
|         scriptEl.onreadystatechange = function () { | ||||
|             handle(); | ||||
|  | ||||
|             scriptEl.onreadystatechange = null; | ||||
|             scriptEl.parentNode.removeChild(scriptEl); | ||||
|             scriptEl = null; | ||||
|         }; | ||||
|         globe.document.documentElement.appendChild(scriptEl); | ||||
|  | ||||
|         return handle; | ||||
|     }; | ||||
| }; | ||||
| }); | ||||
| require.register("calvinmetcalf-setImmediate/lib/timeout.js", function(exports, require, module){ | ||||
| "use strict"; | ||||
| exports.test = function () { | ||||
|     return true; | ||||
| }; | ||||
|  | ||||
| exports.install = function (t) { | ||||
|     return function () { | ||||
|         setTimeout(t, 0); | ||||
|     }; | ||||
| }; | ||||
| }); | ||||
| require.register("calvinmetcalf-setImmediate/lib/global.js", function(exports, require, module){ | ||||
| module.exports = typeof global === "object" && global ? global : this; | ||||
| }); | ||||
| require.register("calvinmetcalf-setImmediate/lib/mutation.js", function(exports, require, module){ | ||||
| "use strict"; | ||||
| //based off rsvp | ||||
| //https://github.com/tildeio/rsvp.js/blob/master/lib/rsvp/async.js | ||||
| var globe = require("./global"); | ||||
|  | ||||
| var MutationObserver = globe.MutationObserver || globe.WebKitMutationObserver; | ||||
|  | ||||
| exports.test = function () { | ||||
|     return MutationObserver; | ||||
| }; | ||||
|  | ||||
| exports.install = function (handle) { | ||||
|     var observer = new MutationObserver(handle); | ||||
|     var element = globe.document.createElement("div"); | ||||
|     observer.observe(element, { attributes: true }); | ||||
|  | ||||
|     // Chrome Memory Leak: https://bugs.webkit.org/show_bug.cgi?id=93661 | ||||
|     globe.addEventListener("unload", function () { | ||||
|         observer.disconnect(); | ||||
|         observer = null; | ||||
|     }, false); | ||||
|     return function () { | ||||
|         element.setAttribute("drainQueue", "drainQueue"); | ||||
|     }; | ||||
| }; | ||||
| }); | ||||
| require.register("lie/lie.js", function(exports, require, module){ | ||||
| var immediate = require('immediate'); | ||||
| // Creates a deferred: an object with a promise and corresponding resolve/reject methods | ||||
| function Promise(resolver) { | ||||
|      if (!(this instanceof Promise)) { | ||||
|         return new Promise(resolver); | ||||
|     } | ||||
|     var queue = []; | ||||
|     var resolved = false; | ||||
|     // The `handler` variable points to the function that will | ||||
|     // 1) handle a .then(onFulfilled, onRejected) call | ||||
|     // 2) handle a .resolve or .reject call (if not fulfilled) | ||||
|     // Before 2), `handler` holds a queue of callbacks. | ||||
|     // After 2), `handler` is a simple .then handler. | ||||
|     // We use only one function to save memory and complexity. | ||||
|      // Case 1) handle a .then(onFulfilled, onRejected) call | ||||
|     function pending(onFulfilled, onRejected){ | ||||
|         return Promise(function(resolver,rejecter){ | ||||
|             queue.push({ | ||||
|                 resolve: onFulfilled, | ||||
|                 reject: onRejected, | ||||
|                 resolver:resolver, | ||||
|                 rejecter:rejecter | ||||
|             }); | ||||
|         }); | ||||
|     } | ||||
|     function then(onFulfilled, onRejected) { | ||||
|         return resolved?resolved(onFulfilled, onRejected):pending(onFulfilled, onRejected); | ||||
|     } | ||||
|     // Case 2) handle a .resolve or .reject call | ||||
|         // (`onFulfilled` acts as a sentinel) | ||||
|         // The actual function signature is | ||||
|         // .re[ject|solve](sentinel, success, value) | ||||
|     function resolve(success, value){ | ||||
|         var action = success ? 'resolve' : 'reject'; | ||||
|         var queued; | ||||
|         var callback; | ||||
|         for (var i = 0, l = queue.length; i < l; i++) { | ||||
|             queued = queue[i]; | ||||
|             callback = queued[action]; | ||||
|             if (typeof callback === 'function') { | ||||
|                 immediate(execute,callback, value, queued.resolver, queued.rejecter); | ||||
|             }else if(success){ | ||||
|                 queued.resolver(value); | ||||
|             }else{ | ||||
|                 queued.rejecter(value); | ||||
|             } | ||||
|         } | ||||
|         // Replace this handler with a simple resolved or rejected handler | ||||
|         resolved = createHandler(then, value, success); | ||||
|     } | ||||
|     this.then = then; | ||||
|     function yes(value) { | ||||
|         if (!resolved) { | ||||
|             resolve(true, value); | ||||
|         } | ||||
|     } | ||||
|     function no (reason) { | ||||
|         if (!resolved) { | ||||
|             resolve(false, reason); | ||||
|         } | ||||
|     } | ||||
|     try{ | ||||
|         resolver(function(a){ | ||||
|             if(a && typeof a.then==='function'){ | ||||
|                 a.then(yes,no); | ||||
|             }else{ | ||||
|                 yes(a); | ||||
|             } | ||||
|         },no); | ||||
|     }catch(e){ | ||||
|         no(e); | ||||
|     } | ||||
| } | ||||
|  | ||||
| // Creates a fulfilled or rejected .then function | ||||
| function createHandler(then, value, success) { | ||||
|     return function(onFulfilled, onRejected) { | ||||
|         var callback = success ? onFulfilled : onRejected; | ||||
|         if (typeof callback !== 'function') { | ||||
|             return Promise(function(resolve,reject){ | ||||
|                 then(resolve,reject); | ||||
|             }); | ||||
|         } | ||||
|         return Promise(function(resolve,reject){ | ||||
|             immediate(execute,callback,value,resolve,reject); | ||||
|        }); | ||||
|     }; | ||||
| } | ||||
|  | ||||
| // Executes the callback with the specified value, | ||||
| // resolving or rejecting the deferred | ||||
| function execute(callback, value, resolve, reject) { | ||||
|         try { | ||||
|             var result = callback(value); | ||||
|             if (result && typeof result.then === 'function') { | ||||
|                 result.then(resolve, reject); | ||||
|             } | ||||
|             else { | ||||
|                 resolve(result); | ||||
|             } | ||||
|         } | ||||
|         catch (error) { | ||||
|             reject(error); | ||||
|         } | ||||
| } | ||||
| module.exports = Promise; | ||||
|  | ||||
| }); | ||||
| require.alias("calvinmetcalf-setImmediate/lib/index.js", "lie/deps/immediate/lib/index.js"); | ||||
| require.alias("calvinmetcalf-setImmediate/lib/nextTick.js", "lie/deps/immediate/lib/nextTick.js"); | ||||
| require.alias("calvinmetcalf-setImmediate/lib/postMessage.js", "lie/deps/immediate/lib/postMessage.js"); | ||||
| require.alias("calvinmetcalf-setImmediate/lib/messageChannel.js", "lie/deps/immediate/lib/messageChannel.js"); | ||||
| require.alias("calvinmetcalf-setImmediate/lib/stateChange.js", "lie/deps/immediate/lib/stateChange.js"); | ||||
| require.alias("calvinmetcalf-setImmediate/lib/timeout.js", "lie/deps/immediate/lib/timeout.js"); | ||||
| require.alias("calvinmetcalf-setImmediate/lib/global.js", "lie/deps/immediate/lib/global.js"); | ||||
| require.alias("calvinmetcalf-setImmediate/lib/mutation.js", "lie/deps/immediate/lib/mutation.js"); | ||||
| require.alias("calvinmetcalf-setImmediate/lib/index.js", "lie/deps/immediate/index.js"); | ||||
| require.alias("calvinmetcalf-setImmediate/lib/index.js", "immediate/index.js"); | ||||
| require.alias("calvinmetcalf-setImmediate/lib/index.js", "calvinmetcalf-setImmediate/index.js"); | ||||
|  | ||||
| require.alias("lie/lie.js", "lie/index.js"); | ||||
|  | ||||
| L.Util.Promise = require("lie"); | ||||
| })(); | ||||
|  | ||||
| L.Util.ajax = function(url, options) { | ||||
| 	'use strict'; | ||||
| 	options = options || {}; | ||||
| 	if (options.jsonp) { | ||||
| 		return L.Util.ajax.jsonp(url, options); | ||||
| 	} | ||||
| 	var request; | ||||
| 	var cancel; | ||||
| 	var out = L.Util.Promise(function(resolve,reject){ | ||||
| 		var Ajax; | ||||
| 		cancel=reject; | ||||
| 		// the following is from JavaScript: The Definitive Guide | ||||
| 		if (window.XMLHttpRequest === undefined) { | ||||
| 			Ajax = function() { | ||||
| 				try { | ||||
| 					return new ActiveXObject('Microsoft.XMLHTTP.6.0'); | ||||
| 				} | ||||
| 				catch (e1) { | ||||
| 					try { | ||||
| 						return new ActiveXObject('Microsoft.XMLHTTP.3.0'); | ||||
| 					} | ||||
| 					catch (e2) { | ||||
| 						reject('XMLHttpRequest is not supported'); | ||||
| 					} | ||||
| 				} | ||||
| 			}; | ||||
| 		} | ||||
| 		else { | ||||
| 			Ajax = window.XMLHttpRequest; | ||||
| 		} | ||||
| 		var response; | ||||
| 		request = new Ajax(); | ||||
| 		request.open('GET', url); | ||||
| 		request.onreadystatechange = function() { | ||||
| 			/*jslint evil: true */ | ||||
| 			if (request.readyState === 4) { | ||||
| 				if((request.status < 400&&options.local)|| request.status===200) { | ||||
| 					if (window.JSON) { | ||||
| 						response = JSON.parse(request.responseText); | ||||
| 					} else if (options.evil) { | ||||
| 						response = eval('(' + request.responseText + ')'); | ||||
| 					} | ||||
| 					resolve(response); | ||||
| 				} else { | ||||
| 					if(!request.status){ | ||||
| 						reject('Attempted cross origin request without CORS enabled'); | ||||
| 					}else{ | ||||
| 						reject(request.statusText); | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		}; | ||||
| 		request.send(); | ||||
| 	}); | ||||
| 	out.then(null,function(reason){ | ||||
| 		request.abort(); | ||||
| 		return reason; | ||||
| 	}); | ||||
| 	out.abort = cancel; | ||||
| 	return out; | ||||
| }; | ||||
|  | ||||
| L.Util.jsonp = function(url, options) { | ||||
| 	options = options || {}; | ||||
| 	var head = document.getElementsByTagName('head')[0]; | ||||
| 	var scriptNode = L.DomUtil.create('script', '', head); | ||||
| 	var cbName, ourl, cbSuffix, cancel; | ||||
| 	var out = L.Util.Promise(function(resolve, reject){ | ||||
| 		cancel=reject; | ||||
| 		var cbParam = options.cbParam || 'callback'; | ||||
| 		if (options.callbackName) { | ||||
| 			cbName = options.callbackName; | ||||
| 		} | ||||
| 		else { | ||||
| 			cbSuffix = '_' + ('' + Math.random()).slice(2); | ||||
| 			cbName = 'L.Util.jsonp.cb.' + cbSuffix; | ||||
| 		} | ||||
| 		scriptNode.type = 'text/javascript'; | ||||
| 		if (cbSuffix) { | ||||
| 			L.Util.jsonp.cb[cbSuffix] = function(data) { | ||||
| 				head.removeChild(scriptNode); | ||||
| 				delete L.Util.jsonp.cb[cbSuffix]; | ||||
| 				resolve(data); | ||||
| 			}; | ||||
| 		} | ||||
| 		if (url.indexOf('?') === -1) { | ||||
| 			ourl = url + '?' + cbParam + '=' + cbName; | ||||
| 		} | ||||
| 		else { | ||||
| 			ourl = url + '&' + cbParam + '=' + cbName; | ||||
| 		} | ||||
| 		scriptNode.src = ourl; | ||||
| 	}).then(null,function(reason){ | ||||
| 	    head.removeChild(scriptNode); | ||||
| 		delete L.Util.ajax.cb[cbSuffix]; | ||||
| 		return reason; | ||||
| 	}); | ||||
| 	out.abort = cancel; | ||||
| 	return out; | ||||
| }; | ||||
| L.Util.jsonp.cb = {}; | ||||
|  | ||||
| L.GeoJSON.AJAX = L.GeoJSON.extend({ | ||||
| 	defaultAJAXparams: { | ||||
| 		dataType: 'json', | ||||
| 		callbackParam: 'callback', | ||||
| 		local:false, | ||||
| 		middleware: function(f) { | ||||
| 			return f; | ||||
| 		} | ||||
| 	}, | ||||
| 	initialize: function(url, options) { | ||||
|  | ||||
| 		this.urls = []; | ||||
| 		if (url) { | ||||
| 			if (typeof url === 'string') { | ||||
| 				this.urls.push(url); | ||||
| 			} | ||||
| 			else if (typeof url.pop === 'function') { | ||||
| 				this.urls = this.urls.concat(url); | ||||
| 			} | ||||
| 			else { | ||||
| 				options = url; | ||||
| 				url = undefined; | ||||
| 			} | ||||
| 		} | ||||
| 		var ajaxParams = L.Util.extend({}, this.defaultAJAXparams); | ||||
|  | ||||
| 		for (var i in options) { | ||||
| 			if (this.defaultAJAXparams.hasOwnProperty(i)) { | ||||
| 				ajaxParams[i] = options[i]; | ||||
| 			} | ||||
| 		} | ||||
| 		this.ajaxParams = ajaxParams; | ||||
| 		this._layers = {}; | ||||
| 		L.Util.setOptions(this, options); | ||||
| 		this.on('data:loaded', function() { | ||||
| 			if (this.filter) { | ||||
| 				this.refilter(this.filter); | ||||
| 			} | ||||
| 		}, this); | ||||
| 		var self = this; | ||||
| 		if (this.urls.length > 0) { | ||||
| 			L.Util.Promise(function(yes){ | ||||
| 			    yes(); | ||||
| 			}).then(function(){ | ||||
| 			    self.addUrl(); | ||||
| 		    }); | ||||
| 		} | ||||
| 	}, | ||||
| 	clearLayers: function() { | ||||
| 		this.urls = []; | ||||
| 		L.GeoJSON.prototype.clearLayers.call(this); | ||||
| 		return this; | ||||
| 	}, | ||||
| 	addUrl: function(url) { | ||||
| 		var self = this; | ||||
| 		if (url) { | ||||
| 			if (typeof url === 'string') { | ||||
| 				self.urls.push(url); | ||||
| 			} | ||||
| 			else if (typeof url.pop === 'function') { | ||||
| 				self.urls = self.urls.concat(url); | ||||
| 			} | ||||
| 		} | ||||
| 		var loading = self.urls.length; | ||||
| 		var done = 0; | ||||
| 		self.fire('data:loading'); | ||||
| 		self.urls.forEach(function(url) { | ||||
| 			if (self.ajaxParams.dataType.toLowerCase() === 'json') { | ||||
| 				L.Util.ajax(url,self.ajaxParams).then(function(d) { | ||||
| 					var data = self.ajaxParams.middleware(d); | ||||
| 					self.addData(data); | ||||
| 					self.fire('data:progress',data); | ||||
| 				},function(err){ | ||||
| 					self.fire('data:progress',{error:err}); | ||||
| 				}); | ||||
| 			} | ||||
| 			else if (self.ajaxParams.dataType.toLowerCase() === 'jsonp') { | ||||
| 				L.Util.jsonp(url,self.ajaxParams).then(function(d) { | ||||
| 					var data = self.ajaxParams.middleware(d); | ||||
| 					self.addData(data); | ||||
| 					self.fire('data:progress',data); | ||||
| 				},function(err){ | ||||
| 					self.fire('data:progress',{error:err}); | ||||
| 				}); | ||||
| 			} | ||||
| 		}); | ||||
| 		self.on('data:progress', function() { | ||||
| 			if (++done === loading) { | ||||
| 				self.fire('data:loaded'); | ||||
| 			} | ||||
| 		}); | ||||
| 	}, | ||||
| 	refresh: function(url) { | ||||
| 		url = url || this.urls; | ||||
| 		this.clearLayers(); | ||||
| 		this.addUrl(url); | ||||
| 	}, | ||||
| 	refilter: function(func) { | ||||
| 		if (typeof func !== 'function') { | ||||
| 			this.filter = false; | ||||
| 			this.eachLayer(function(a) { | ||||
| 				a.setStyle({ | ||||
| 					stroke: true, | ||||
| 					clickable: true | ||||
| 				}); | ||||
| 			}); | ||||
| 		} | ||||
| 		else { | ||||
| 			this.filter = func; | ||||
| 			this.eachLayer(function(a) { | ||||
|  | ||||
| 				if (func(a.feature)) { | ||||
| 					a.setStyle({ | ||||
| 						stroke: true, | ||||
| 						clickable: true | ||||
| 					}); | ||||
| 				} | ||||
| 				else { | ||||
| 					a.setStyle({ | ||||
| 						stroke: false, | ||||
| 						clickable: false | ||||
| 					}); | ||||
| 				} | ||||
| 			}); | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| L.geoJson.ajax = function(geojson, options) { | ||||
| 	return new L.GeoJSON.AJAX(geojson, options); | ||||
| }; | ||||
| @@ -1,125 +0,0 @@ | ||||
| /* | ||||
|   Leaflet.AwesomeMarkers, a plugin that adds colorful iconic markers for Leaflet, based on the Font Awesome icons | ||||
|   (c) 2012-2013, Lennard Voogdt | ||||
|  | ||||
|   http://leafletjs.com | ||||
|   https://github.com/lvoogdt | ||||
| */ | ||||
|  | ||||
| /*global L*/ | ||||
|  | ||||
| (function (window, document, undefined) { | ||||
|     "use strict"; | ||||
|     /* | ||||
|      * Leaflet.AwesomeMarkers assumes that you have already included the Leaflet library. | ||||
|      */ | ||||
|  | ||||
|     L.AwesomeMarkers = {}; | ||||
|  | ||||
|     L.AwesomeMarkers.version = '2.0.1'; | ||||
|  | ||||
|     L.AwesomeMarkers.Icon = L.Icon.extend({ | ||||
|         options: { | ||||
|             iconSize: [35, 45], | ||||
|             iconAnchor:   [17, 42], | ||||
|             popupAnchor: [1, -32], | ||||
|             shadowAnchor: [10, 12], | ||||
|             shadowSize: [36, 16], | ||||
|             className: 'awesome-marker', | ||||
|             prefix: 'glyphicon', | ||||
|             spinClass: 'fa-spin', | ||||
|             extraClasses: '', | ||||
|             icon: 'home', | ||||
|             markerColor: 'blue', | ||||
|             iconColor: 'white' | ||||
|         }, | ||||
|  | ||||
|         initialize: function (options) { | ||||
|             options = L.Util.setOptions(this, options); | ||||
|         }, | ||||
|  | ||||
|         createIcon: function () { | ||||
|             var div = document.createElement('div'), | ||||
|                 options = this.options; | ||||
|  | ||||
|             if (options.icon) { | ||||
|                 div.innerHTML = this._createInner(); | ||||
|             } | ||||
|  | ||||
|             if (options.bgPos) { | ||||
|                 div.style.backgroundPosition = | ||||
|                     (-options.bgPos.x) + 'px ' + (-options.bgPos.y) + 'px'; | ||||
|             } | ||||
|  | ||||
|             this._setIconStyles(div, 'icon-' + options.markerColor); | ||||
|             return div; | ||||
|         }, | ||||
|  | ||||
|         _createInner: function() { | ||||
|             var iconClass, iconSpinClass = "", iconColorClass = "", iconColorStyle = "", options = this.options; | ||||
|  | ||||
|             if(options.icon.slice(0,options.prefix.length+1) === options.prefix + "-") { | ||||
|                 iconClass = options.icon; | ||||
|             } else { | ||||
|                 iconClass = options.prefix + "-" + options.icon; | ||||
|             } | ||||
|  | ||||
|             if(options.spin && typeof options.spinClass === "string") { | ||||
|                 iconSpinClass = options.spinClass; | ||||
|             } | ||||
|  | ||||
|             if(options.iconColor) { | ||||
|                 if(options.iconColor === 'white' || options.iconColor === 'black') { | ||||
|                     iconColorClass = "icon-" + options.iconColor; | ||||
|                 } else { | ||||
|                     iconColorStyle = "style='color: " + options.iconColor + "' "; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             return "<i " + iconColorStyle + "class='" + options.extraClasses + " " + options.prefix + " " + iconClass + " " + iconSpinClass + " " + iconColorClass + "'></i>"; | ||||
|         }, | ||||
|  | ||||
|         _setIconStyles: function (img, name) { | ||||
|             var options = this.options, | ||||
|                 size = L.point(options[name === 'shadow' ? 'shadowSize' : 'iconSize']), | ||||
|                 anchor; | ||||
|  | ||||
|             if (name === 'shadow') { | ||||
|                 anchor = L.point(options.shadowAnchor || options.iconAnchor); | ||||
|             } else { | ||||
|                 anchor = L.point(options.iconAnchor); | ||||
|             } | ||||
|  | ||||
|             if (!anchor && size) { | ||||
|                 anchor = size.divideBy(2, true); | ||||
|             } | ||||
|  | ||||
|             img.className = 'awesome-marker-' + name + ' ' + options.className; | ||||
|  | ||||
|             if (anchor) { | ||||
|                 img.style.marginLeft = (-anchor.x) + 'px'; | ||||
|                 img.style.marginTop  = (-anchor.y) + 'px'; | ||||
|             } | ||||
|  | ||||
|             if (size) { | ||||
|                 img.style.width  = size.x + 'px'; | ||||
|                 img.style.height = size.y + 'px'; | ||||
|             } | ||||
|         }, | ||||
|  | ||||
|         createShadow: function () { | ||||
|             var div = document.createElement('div'); | ||||
|  | ||||
|             this._setIconStyles(div, 'shadow'); | ||||
|             return div; | ||||
|       } | ||||
|     }); | ||||
|          | ||||
|     L.AwesomeMarkers.icon = function (options) { | ||||
|         return new L.AwesomeMarkers.Icon(options); | ||||
|     }; | ||||
|  | ||||
| }(this, document)); | ||||
|  | ||||
|  | ||||
|  | ||||
| Before Width: | Height: | Size: 157 B |