mirror of
https://github.com/FoggedLens/iD.git
synced 2026-02-14 09:42:56 +00:00
347 lines
9.5 KiB
JavaScript
347 lines
9.5 KiB
JavaScript
window.iD = function () {
|
|
window.locale.en = iD.data.en;
|
|
window.locale.current('en');
|
|
|
|
var context = {},
|
|
storage;
|
|
|
|
// https://github.com/systemed/iD/issues/772
|
|
// http://mathiasbynens.be/notes/localstorage-pattern#comment-9
|
|
try { storage = localStorage; } catch (e) {}
|
|
storage = storage || (function() {
|
|
var s = {};
|
|
return {
|
|
getItem: function(k) { return s[k]; },
|
|
setItem: function(k, v) { s[k] = v; },
|
|
removeItem: function(k) { delete s[k]; }
|
|
};
|
|
})();
|
|
|
|
context.storage = function(k, v) {
|
|
try {
|
|
if (arguments.length === 1) return storage.getItem(k);
|
|
else if (v === null) storage.removeItem(k);
|
|
else storage.setItem(k, v);
|
|
} catch(e) {
|
|
// localstorage quota exceeded
|
|
/* jshint devel:true */
|
|
if (typeof console !== 'undefined') console.error('localStorage quota exceeded');
|
|
/* jshint devel:false */
|
|
}
|
|
};
|
|
|
|
var history = iD.History(context),
|
|
dispatch = d3.dispatch('enter', 'exit'),
|
|
mode,
|
|
container,
|
|
ui = iD.ui(context),
|
|
connection = iD.Connection(),
|
|
locale = iD.detect().locale,
|
|
localePath;
|
|
|
|
if (locale && iD.data.locales.indexOf(locale) === -1) {
|
|
locale = locale.split('-')[0];
|
|
}
|
|
|
|
connection.on('load.context', function loadContext(err, result) {
|
|
history.merge(result.data, result.extent);
|
|
});
|
|
|
|
context.preauth = function(options) {
|
|
connection.switch(options);
|
|
return context;
|
|
};
|
|
|
|
context.locale = function(_, path) {
|
|
locale = _;
|
|
localePath = path;
|
|
return context;
|
|
};
|
|
|
|
context.loadLocale = function(cb) {
|
|
if (locale && locale !== 'en' && iD.data.locales.indexOf(locale) !== -1) {
|
|
localePath = localePath || context.assetPath() + 'locales/' + locale + '.json';
|
|
d3.json(localePath, function(err, result) {
|
|
window.locale[locale] = result;
|
|
window.locale.current(locale);
|
|
cb();
|
|
});
|
|
} else {
|
|
cb();
|
|
}
|
|
};
|
|
|
|
/* Straight accessors. Avoid using these if you can. */
|
|
context.ui = function() { return ui; };
|
|
context.connection = function() { return connection; };
|
|
context.history = function() { return history; };
|
|
|
|
/* History */
|
|
context.graph = history.graph;
|
|
context.changes = history.changes;
|
|
context.intersects = history.intersects;
|
|
|
|
var inIntro = false;
|
|
|
|
context.inIntro = function(_) {
|
|
if (!arguments.length) return inIntro;
|
|
inIntro = _;
|
|
return context;
|
|
};
|
|
|
|
context.save = function() {
|
|
if (inIntro) return;
|
|
history.save();
|
|
if (history.hasChanges()) return t('save.unsaved_changes');
|
|
};
|
|
|
|
context.flush = function() {
|
|
connection.flush();
|
|
history.reset();
|
|
return context;
|
|
};
|
|
|
|
// Debounce save, since it's a synchronous localStorage write,
|
|
// and history changes can happen frequently (e.g. when dragging).
|
|
var debouncedSave = _.debounce(context.save, 350);
|
|
function withDebouncedSave(fn) {
|
|
return function() {
|
|
var result = fn.apply(history, arguments);
|
|
debouncedSave();
|
|
return result;
|
|
};
|
|
}
|
|
|
|
context.perform = withDebouncedSave(history.perform);
|
|
context.replace = withDebouncedSave(history.replace);
|
|
context.pop = withDebouncedSave(history.pop);
|
|
context.undo = withDebouncedSave(history.undo);
|
|
context.redo = withDebouncedSave(history.redo);
|
|
|
|
/* Graph */
|
|
context.hasEntity = function(id) {
|
|
return history.graph().hasEntity(id);
|
|
};
|
|
|
|
context.entity = function(id) {
|
|
return history.graph().entity(id);
|
|
};
|
|
|
|
context.childNodes = function(way) {
|
|
return history.graph().childNodes(way);
|
|
};
|
|
|
|
context.geometry = function(id) {
|
|
return context.entity(id).geometry(history.graph());
|
|
};
|
|
|
|
/* Modes */
|
|
context.enter = function(newMode) {
|
|
if (mode) {
|
|
mode.exit();
|
|
dispatch.exit(mode);
|
|
}
|
|
|
|
mode = newMode;
|
|
mode.enter();
|
|
dispatch.enter(mode);
|
|
};
|
|
|
|
context.mode = function() {
|
|
return mode;
|
|
};
|
|
|
|
context.selectedIDs = function() {
|
|
if (mode && mode.selectedIDs) {
|
|
return mode.selectedIDs();
|
|
} else {
|
|
return [];
|
|
}
|
|
};
|
|
|
|
context.loadEntity = function(id, zoomTo) {
|
|
if (zoomTo !== false) {
|
|
connection.loadEntity(id, function(error, entity) {
|
|
if (entity) {
|
|
map.zoomTo(entity);
|
|
}
|
|
});
|
|
}
|
|
|
|
map.on('drawn.loadEntity', function() {
|
|
if (!context.hasEntity(id)) return;
|
|
map.on('drawn.loadEntity', null);
|
|
context.on('enter.loadEntity', null);
|
|
context.enter(iD.modes.Select(context, [id]));
|
|
});
|
|
|
|
context.on('enter.loadEntity', function() {
|
|
if (mode.id !== 'browse') {
|
|
map.on('drawn.loadEntity', null);
|
|
context.on('enter.loadEntity', null);
|
|
}
|
|
});
|
|
};
|
|
|
|
context.editable = function() {
|
|
return map.editable() && mode && mode.id !== 'save';
|
|
};
|
|
|
|
/* Behaviors */
|
|
context.install = function(behavior) {
|
|
context.surface().call(behavior);
|
|
};
|
|
|
|
context.uninstall = function(behavior) {
|
|
context.surface().call(behavior.off);
|
|
};
|
|
|
|
/* Projection */
|
|
function rawMercator() {
|
|
var project = d3.geo.mercator.raw,
|
|
k = 512 / Math.PI, // scale
|
|
x = 0, y = 0, // translate
|
|
clipExtent = [[0, 0], [0, 0]];
|
|
|
|
function projection(point) {
|
|
point = project(point[0] * Math.PI / 180, point[1] * Math.PI / 180);
|
|
return [point[0] * k + x, y - point[1] * k];
|
|
}
|
|
|
|
projection.invert = function(point) {
|
|
point = project.invert((point[0] - x) / k, (y - point[1]) / k);
|
|
return point && [point[0] * 180 / Math.PI, point[1] * 180 / Math.PI];
|
|
};
|
|
|
|
projection.scale = function(_) {
|
|
if (!arguments.length) return k;
|
|
k = +_;
|
|
return projection;
|
|
};
|
|
|
|
projection.translate = function(_) {
|
|
if (!arguments.length) return [x, y];
|
|
x = +_[0];
|
|
y = +_[1];
|
|
return projection;
|
|
};
|
|
|
|
projection.clipExtent = function(_) {
|
|
if (!arguments.length) return clipExtent;
|
|
clipExtent = _;
|
|
return projection;
|
|
};
|
|
|
|
projection.stream = d3.geo.transform({
|
|
point: function(x, y) {
|
|
x = projection([x, y]);
|
|
this.stream.point(x[0], x[1]);
|
|
}
|
|
}).stream;
|
|
|
|
return projection;
|
|
}
|
|
|
|
context.projection = rawMercator();
|
|
|
|
/* Background */
|
|
var background = iD.Background(context);
|
|
context.background = function() { return background; };
|
|
|
|
/* Map */
|
|
var map = iD.Map(context);
|
|
context.map = function() { return map; };
|
|
context.layers = function() { return map.layers; };
|
|
context.surface = function() { return map.surface; };
|
|
context.mouse = map.mouse;
|
|
context.extent = map.extent;
|
|
context.pan = map.pan;
|
|
context.zoomIn = map.zoomIn;
|
|
context.zoomOut = map.zoomOut;
|
|
|
|
context.surfaceRect = function() {
|
|
// Work around a bug in Firefox.
|
|
// http://stackoverflow.com/questions/18153989/
|
|
// https://bugzilla.mozilla.org/show_bug.cgi?id=530985
|
|
return context.surface().node().parentNode.getBoundingClientRect();
|
|
};
|
|
|
|
/* Presets */
|
|
var presets = iD.presets()
|
|
.load(iD.data.presets);
|
|
|
|
context.presets = function() {
|
|
return presets;
|
|
};
|
|
|
|
context.container = function(_) {
|
|
if (!arguments.length) return container;
|
|
container = _;
|
|
container.classed('id-container', true);
|
|
return context;
|
|
};
|
|
|
|
var embed = false;
|
|
context.embed = function(_) {
|
|
if (!arguments.length) return embed;
|
|
embed = _;
|
|
return context;
|
|
};
|
|
|
|
var assetPath = '';
|
|
context.assetPath = function(_) {
|
|
if (!arguments.length) return assetPath;
|
|
assetPath = _;
|
|
return context;
|
|
};
|
|
|
|
var assetMap = {};
|
|
context.assetMap = function(_) {
|
|
if (!arguments.length) return assetMap;
|
|
assetMap = _;
|
|
return context;
|
|
};
|
|
|
|
context.imagePath = function(_) {
|
|
var asset = 'img/' + _;
|
|
return assetMap[asset] || assetPath + asset;
|
|
};
|
|
|
|
return d3.rebind(context, dispatch, 'on');
|
|
};
|
|
|
|
iD.version = '1.3.4';
|
|
|
|
(function() {
|
|
var detected = {};
|
|
|
|
var ua = navigator.userAgent,
|
|
msie = new RegExp('MSIE ([0-9]{1,}[\\.0-9]{0,})');
|
|
|
|
if (msie.exec(ua) !== null) {
|
|
var rv = parseFloat(RegExp.$1);
|
|
detected.support = !(rv && rv < 9);
|
|
} else {
|
|
detected.support = true;
|
|
}
|
|
|
|
// Added due to incomplete svg style support. See #715
|
|
detected.opera = ua.indexOf('Opera') >= 0;
|
|
|
|
detected.locale = navigator.language || navigator.userLanguage;
|
|
|
|
detected.filedrop = (window.FileReader && 'ondrop' in window);
|
|
|
|
function nav(x) {
|
|
return navigator.userAgent.indexOf(x) !== -1;
|
|
}
|
|
|
|
if (nav('Win')) detected.os = 'win';
|
|
else if (nav('Mac')) detected.os = 'mac';
|
|
else if (nav('X11')) detected.os = 'linux';
|
|
else if (nav('Linux')) detected.os = 'linux';
|
|
else detected.os = 'win';
|
|
|
|
iD.detect = function() { return detected; };
|
|
})();
|