mirror of
https://github.com/FoggedLens/iD.git
synced 2026-02-13 09:12:52 +00:00
221 lines
6.8 KiB
JavaScript
221 lines
6.8 KiB
JavaScript
iD.Connection = function() {
|
|
|
|
var event = d3.dispatch('auth', 'load'),
|
|
apiURL = 'http://www.openstreetmap.org',
|
|
connection = {},
|
|
refNodes = {},
|
|
user = {},
|
|
apiTilesLoaded = {},
|
|
oauth = iD.OAuth().api(apiURL);
|
|
|
|
// Request data within the bbox from an external OSM server.
|
|
function bboxFromAPI(box, callback) {
|
|
loadFromURL(apiURL + '/api/0.6/map?bbox=' +
|
|
[box[0][0], box[1][1], box[1][0], box[0][1]], callback);
|
|
}
|
|
|
|
function loadFromURL(url, callback) {
|
|
d3.xml(url, function(err, dom) { callback(err, parse(dom)); });
|
|
}
|
|
|
|
function getNodes(obj) {
|
|
var nodes = [], nelems = obj.getElementsByTagName('nd');
|
|
for (var i = 0, l = nelems.length; i < l; i++) {
|
|
nodes[i] = 'n' + nelems[i].attributes.ref.nodeValue;
|
|
refNodes['n' + nelems[i].attributes.ref.nodeValue] = true;
|
|
}
|
|
return nodes;
|
|
}
|
|
|
|
function getTags(obj) {
|
|
var tags = {}, tagelems = obj.getElementsByTagName('tag');
|
|
for (var i = 0, l = tagelems.length; i < l; i++) {
|
|
var item = tagelems[i];
|
|
tags[item.attributes.k.nodeValue] = item.attributes.v.nodeValue;
|
|
}
|
|
return tags;
|
|
}
|
|
|
|
function getMembers(obj) {
|
|
var members = [],
|
|
elems = obj.getElementsByTagName('member');
|
|
|
|
for (var i = 0, l = elems.length; i < l; i++) {
|
|
members[i] = {
|
|
id: elems[i].attributes.type.nodeValue[0] + elems[i].attributes.ref.nodeValue,
|
|
type: elems[i].attributes.type.nodeValue,
|
|
role: elems[i].attributes.role.nodeValue
|
|
};
|
|
}
|
|
return members;
|
|
}
|
|
|
|
function objectData(obj) {
|
|
var o = {
|
|
type: obj.nodeName,
|
|
members: getMembers(obj),
|
|
nodes: getNodes(obj),
|
|
tags: getTags(obj)
|
|
};
|
|
for (var i = 0, l = obj.attributes.length; i < l; i++) {
|
|
var n = obj.attributes[i].nodeName;
|
|
var v = obj.attributes[i].nodeValue;
|
|
o[n] = v;
|
|
}
|
|
if (o.lat) o.lat = parseFloat(o.lat);
|
|
if (o.lon) o.lon = parseFloat(o.lon);
|
|
o._id = o.id;
|
|
o.id = o.type[0] + o.id;
|
|
return iD.Entity(o);
|
|
}
|
|
|
|
function parse(dom) {
|
|
if (!dom.childNodes) return new Error('Bad request');
|
|
var root = dom.childNodes[0];
|
|
var entities = {};
|
|
refNodes = {};
|
|
function addEntity(obj) {
|
|
var o = objectData(obj);
|
|
if (o.type === 'node') o._poi = !refNodes[o.id];
|
|
entities[o.id] = o;
|
|
}
|
|
|
|
_.forEach(root.getElementsByTagName('way'), addEntity);
|
|
_.forEach(root.getElementsByTagName('node'), addEntity);
|
|
_.forEach(root.getElementsByTagName('relation'), addEntity);
|
|
|
|
return iD.Graph(entities);
|
|
}
|
|
|
|
function authenticate(callback) {
|
|
return oauth.authenticate(function() {
|
|
event.auth();
|
|
if (callback) callback();
|
|
});
|
|
}
|
|
|
|
function authenticated() {
|
|
return oauth.authenticated();
|
|
}
|
|
|
|
function putChangeset(changes, comment, callback) {
|
|
oauth.xhr({
|
|
method: 'PUT',
|
|
path: '/api/0.6/changeset/create',
|
|
options: { header: { 'Content-Type': 'text/xml' } },
|
|
content: iD.format.XML.changeset(comment)
|
|
},
|
|
function (changeset_id) {
|
|
oauth.xhr({
|
|
method: 'POST',
|
|
path: '/api/0.6/changeset/' + changeset_id + '/upload',
|
|
options: { header: { 'Content-Type': 'text/xml' } },
|
|
content: iD.format.XML.osmChange(user.id, changeset_id, changes)
|
|
}, function () {
|
|
oauth.xhr({
|
|
method: 'PUT',
|
|
path: '/api/0.6/changeset/' + changeset_id + '/close'
|
|
}, function () {
|
|
callback(changeset_id);
|
|
});
|
|
});
|
|
});
|
|
}
|
|
|
|
function userDetails(callback) {
|
|
oauth.xhr({ method: 'GET', path: '/api/0.6/user/details' }, function(user_details) {
|
|
var u = user_details.getElementsByTagName('user')[0];
|
|
callback(connection.user({
|
|
display_name: u.attributes.display_name.nodeValue,
|
|
id: u.attributes.id.nodeValue
|
|
}).user());
|
|
});
|
|
}
|
|
|
|
function tileAtZoom(t, distance) {
|
|
var power = Math.pow(2, distance);
|
|
return [
|
|
Math.floor(t[0] * power),
|
|
Math.floor(t[1] * power),
|
|
t[2] + distance];
|
|
}
|
|
|
|
function tileAlreadyLoaded(c) {
|
|
if (apiTilesLoaded[c]) return false;
|
|
for (var i = 0; i < 4; i++) {
|
|
if (apiTilesLoaded[tileAtZoom(c, -i)]) return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
function apiRequestExtent(extent) {
|
|
bboxFromAPI(extent, event.load);
|
|
}
|
|
|
|
function loadTiles(projection) {
|
|
var scaleExtent = [16, 16],
|
|
s = projection.scale(),
|
|
tiles = d3.geo.tile()
|
|
.scaleExtent(scaleExtent)
|
|
.scale(s)
|
|
.translate(projection.translate())(),
|
|
z = Math.max(Math.log(s) / Math.log(2) - 8, 0),
|
|
rz = Math.max(scaleExtent[0], Math.min(scaleExtent[1], Math.floor(z))),
|
|
ts = 256 * Math.pow(2, z - rz),
|
|
tile_origin = [
|
|
s / 2 - projection.translate()[0],
|
|
s / 2 - projection.translate()[1]];
|
|
|
|
function apiExtentBox(c) {
|
|
var x = (c[0] * ts) - tile_origin[0];
|
|
var y = (c[1] * ts) - tile_origin[1];
|
|
apiTilesLoaded[c] = true;
|
|
return [
|
|
projection.invert([x, y]),
|
|
projection.invert([x + ts, y + ts])];
|
|
}
|
|
|
|
return tiles
|
|
.filter(tileAlreadyLoaded)
|
|
.map(apiExtentBox)
|
|
.map(apiRequestExtent);
|
|
}
|
|
|
|
connection.url = function(_) {
|
|
if (!arguments.length) return apiURL;
|
|
apiURL = _;
|
|
oauth.api(_);
|
|
return connection;
|
|
};
|
|
|
|
connection.user = function(_) {
|
|
if (!arguments.length) return user;
|
|
user = _;
|
|
return connection;
|
|
};
|
|
|
|
connection.flush = function() {
|
|
apiTilesLoaded = {};
|
|
return connection;
|
|
};
|
|
|
|
connection.logout = function() {
|
|
oauth.logout();
|
|
event.auth();
|
|
return connection;
|
|
};
|
|
|
|
connection.bboxFromAPI = bboxFromAPI;
|
|
connection.loadFromURL = loadFromURL;
|
|
connection.loadTiles = _.debounce(loadTiles, 1000);
|
|
connection.userDetails = userDetails;
|
|
connection.authenticate = authenticate;
|
|
connection.authenticated = authenticated;
|
|
connection.putChangeset = putChangeset;
|
|
|
|
connection.objectData = objectData;
|
|
connection.apiURL = apiURL;
|
|
|
|
return d3.rebind(connection, event, 'on');
|
|
};
|