29b9d47e79a9ac2477affadac0c8755dc8ce8c23
[rails.git] / vendor / assets / leaflet / leaflet.osm.js
1 L.OSM = {};
2
3 L.OSM.TileLayer = L.TileLayer.extend({
4   options: {
5     url: document.location.protocol === 'https:' ?
6       'https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png' :
7       'http://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png',
8     attribution: '© <a href="http://www.openstreetmap.org/copyright" target="_blank">OpenStreetMap</a> contributors'
9   },
10
11   initialize: function (options) {
12     options = L.Util.setOptions(this, options);
13     L.TileLayer.prototype.initialize.call(this, options.url);
14   }
15 });
16
17 L.OSM.Mapnik = L.OSM.TileLayer.extend({
18   options: {
19     url: document.location.protocol === 'https:' ?
20       'https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png' :
21       'http://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png',
22     maxZoom: 19
23   }
24 });
25
26 L.OSM.CycleMap = L.OSM.TileLayer.extend({
27   options: {
28     url: document.location.protocol === 'https:' ?
29       'https://{s}.tile.thunderforest.com/cycle/{z}/{x}/{y}.png' :
30       'http://{s}.tile.thunderforest.com/cycle/{z}/{x}/{y}.png',
31     attribution: '© <a href="http://www.openstreetmap.org/copyright" target="_blank">OpenStreetMap</a> contributors. Tiles courtesy of <a href="http://www.thunderforest.com/" target="_blank">Andy Allan</a>'
32   }
33 });
34
35 L.OSM.TransportMap = L.OSM.TileLayer.extend({
36   options: {
37     url:  document.location.protocol === 'https:' ?
38       'https://{s}.tile.thunderforest.com/transport/{z}/{x}/{y}.png' :
39       'http://{s}.tile.thunderforest.com/transport/{z}/{x}/{y}.png',
40     attribution: '© <a href="http://www.openstreetmap.org/copyright" target="_blank">OpenStreetMap</a> contributors. Tiles courtesy of <a href="http://www.thunderforest.com/" target="_blank">Andy Allan</a>'
41   }
42 });
43
44 L.OSM.MapQuestOpen = L.OSM.TileLayer.extend({
45   options: {
46     url: document.location.protocol === 'https:' ?
47       'https://otile{s}-s.mqcdn.com/tiles/1.0.0/osm/{z}/{x}/{y}.png' :
48       'http://otile{s}.mqcdn.com/tiles/1.0.0/osm/{z}/{x}/{y}.png',
49     subdomains: '1234',
50     attribution: '© <a href="http://www.openstreetmap.org/copyright" target="_blank">OpenStreetMap</a> contributors. ' + document.location.protocol === 'https:' ?
51       'Tiles courtesy of <a href="http://www.mapquest.com/" target="_blank">MapQuest</a> <img src="https://developer.mapquest.com/content/osm/mq_logo.png">' :
52       'Tiles courtesy of <a href="http://www.mapquest.com/" target="_blank">MapQuest</a> <img src="http://developer.mapquest.com/content/osm/mq_logo.png">'
53   }
54 });
55
56 L.OSM.HOT = L.OSM.TileLayer.extend({
57   options: {
58     url: 'http://{s}.tile.openstreetmap.fr/hot/{z}/{x}/{y}.png',
59     maxZoom: 20,
60     subdomains: 'abc',
61     attribution: '© <a href="http://www.openstreetmap.org/copyright" target="_blank">OpenStreetMap</a> contributors. Tiles courtesy of <a href="http://hot.openstreetmap.org/" target="_blank">Humanitarian OpenStreetMap Team</a>'
62   }
63 });
64
65 L.OSM.DataLayer = L.FeatureGroup.extend({
66   options: {
67     areaTags: ['area', 'building', 'leisure', 'tourism', 'ruins', 'historic', 'landuse', 'military', 'natural', 'sport'],
68     uninterestingTags: ['source', 'source_ref', 'source:ref', 'history', 'attribution', 'created_by', 'tiger:county', 'tiger:tlid', 'tiger:upload_uuid'],
69     styles: {}
70   },
71
72   initialize: function (xml, options) {
73     L.Util.setOptions(this, options);
74
75     L.FeatureGroup.prototype.initialize.call(this);
76
77     if (xml) {
78       this.addData(xml);
79     }
80   },
81
82   addData: function (features) {
83     if (!(features instanceof Array)) {
84       features = this.buildFeatures(features);
85     }
86
87     for (var i = 0; i < features.length; i++) {
88       var feature = features[i], layer;
89
90       if (feature.type === "changeset") {
91         layer = L.rectangle(feature.latLngBounds, this.options.styles.changeset);
92       } else if (feature.type === "node") {
93         layer = L.circleMarker(feature.latLng, this.options.styles.node);
94       } else {
95         var latLngs = new Array(feature.nodes.length);
96
97         for (var j = 0; j < feature.nodes.length; j++) {
98           latLngs[j] = feature.nodes[j].latLng;
99         }
100
101         if (this.isWayArea(feature)) {
102           latLngs.pop(); // Remove last == first.
103           layer = L.polygon(latLngs, this.options.styles.area);
104         } else {
105           layer = L.polyline(latLngs, this.options.styles.way);
106         }
107       }
108
109       layer.addTo(this);
110       layer.feature = feature;
111     }
112   },
113
114   buildFeatures: function (xml) {
115     var features = L.OSM.getChangesets(xml),
116       nodes = L.OSM.getNodes(xml),
117       ways = L.OSM.getWays(xml, nodes),
118       relations = L.OSM.getRelations(xml, nodes, ways);
119
120     for (var node_id in nodes) {
121       var node = nodes[node_id];
122       if (this.interestingNode(node, ways, relations)) {
123         features.push(node);
124       }
125     }
126
127     for (var i = 0; i < ways.length; i++) {
128       var way = ways[i];
129       features.push(way);
130     }
131
132     return features;
133   },
134
135   isWayArea: function (way) {
136     if (way.nodes[0] != way.nodes[way.nodes.length - 1]) {
137       return false;
138     }
139
140     for (var key in way.tags) {
141       if (~this.options.areaTags.indexOf(key)) {
142         return true;
143       }
144     }
145
146     return false;
147   },
148
149   interestingNode: function (node, ways, relations) {
150     var used = false;
151
152     for (var i = 0; i < ways.length; i++) {
153       if (ways[i].nodes.indexOf(node) >= 0) {
154         used = true;
155         break;
156       }
157     }
158
159     if (!used) {
160       return true;
161     }
162
163     for (var i = 0; i < relations.length; i++) {
164       if (relations[i].members.indexOf(node) >= 0)
165         return true;
166     }
167
168     for (var key in node.tags) {
169       if (this.options.uninterestingTags.indexOf(key) < 0) {
170         return true;
171       }
172     }
173
174     return false;
175   }
176 });
177
178 L.Util.extend(L.OSM, {
179   getChangesets: function (xml) {
180     var result = [];
181
182     var nodes = xml.getElementsByTagName("changeset");
183     for (var i = 0; i < nodes.length; i++) {
184       var node = nodes[i], id = node.getAttribute("id");
185       result.push({
186         id: id,
187         type: "changeset",
188         latLngBounds: L.latLngBounds(
189           [node.getAttribute("min_lat"), node.getAttribute("min_lon")],
190           [node.getAttribute("max_lat"), node.getAttribute("max_lon")]),
191         tags: this.getTags(node)
192       });
193     }
194
195     return result;
196   },
197
198   getNodes: function (xml) {
199     var result = {};
200
201     var nodes = xml.getElementsByTagName("node");
202     for (var i = 0; i < nodes.length; i++) {
203       var node = nodes[i], id = node.getAttribute("id");
204       result[id] = {
205         id: id,
206         type: "node",
207         latLng: L.latLng(node.getAttribute("lat"),
208                          node.getAttribute("lon"),
209                          true),
210         tags: this.getTags(node)
211       };
212     }
213
214     return result;
215   },
216
217   getWays: function (xml, nodes) {
218     var result = [];
219
220     var ways = xml.getElementsByTagName("way");
221     for (var i = 0; i < ways.length; i++) {
222       var way = ways[i], nds = way.getElementsByTagName("nd");
223
224       var way_object = {
225         id: way.getAttribute("id"),
226         type: "way",
227         nodes: new Array(nds.length),
228         tags: this.getTags(way)
229       };
230
231       for (var j = 0; j < nds.length; j++) {
232         way_object.nodes[j] = nodes[nds[j].getAttribute("ref")];
233       }
234
235       result.push(way_object);
236     }
237
238     return result;
239   },
240
241   getRelations: function (xml, nodes, ways) {
242     var result = [];
243
244     var rels = xml.getElementsByTagName("relation");
245     for (var i = 0; i < rels.length; i++) {
246       var rel = rels[i], members = rel.getElementsByTagName("member");
247
248       var rel_object = {
249         id: rel.getAttribute("id"),
250         type: "relation",
251         members: new Array(members.length),
252         tags: this.getTags(rel)
253       };
254
255       for (var j = 0; j < members.length; j++) {
256         if (members[j].getAttribute("type") === "node")
257           rel_object.members[j] = nodes[members[j].getAttribute("ref")];
258         else // relation-way and relation-relation membership not implemented
259           rel_object.members[j] = null;
260       }
261
262       result.push(rel_object);
263     }
264
265     return result;
266   },
267
268   getTags: function (xml) {
269     var result = {};
270
271     var tags = xml.getElementsByTagName("tag");
272     for (var j = 0; j < tags.length; j++) {
273       result[tags[j].getAttribute("k")] = tags[j].getAttribute("v");
274     }
275
276     return result;
277   }
278 });