mirror of
https://github.com/FoggedLens/iD.git
synced 2026-05-22 16:19:48 +02:00
f1f6656269
* instead of passing small deltas to the move action and accumulating errors, always pass the absolute delta since entering move mode * also fix cache to handle circular/self-referencing relations
152 lines
3.8 KiB
JavaScript
152 lines
3.8 KiB
JavaScript
iD.modes.Move = function(context, entityIDs) {
|
|
var mode = {
|
|
id: 'move',
|
|
button: 'browse'
|
|
};
|
|
|
|
var keybinding = d3.keybinding('move'),
|
|
edit = iD.behavior.Edit(context),
|
|
annotation = entityIDs.length === 1 ?
|
|
t('operations.move.annotation.' + context.geometry(entityIDs[0])) :
|
|
t('operations.move.annotation.multiple'),
|
|
cache,
|
|
origin,
|
|
delta,
|
|
nudgeInterval;
|
|
|
|
function clearCache() {
|
|
cache = {
|
|
moving: {},
|
|
startLoc: {}
|
|
};
|
|
}
|
|
|
|
function cacheLocs(ids) {
|
|
_.each(ids, function(id) {
|
|
if (cache.moving[id]) return;
|
|
cache.moving[id] = true;
|
|
|
|
var entity = context.hasEntity(id);
|
|
if (!entity) return;
|
|
|
|
if (entity.type === 'node') {
|
|
cache.startLoc[id] = entity.loc;
|
|
} else if (entity.type === 'way') {
|
|
cacheLocs(entity.nodes);
|
|
} else {
|
|
cacheLocs(_.pluck(entity.members, 'id'));
|
|
}
|
|
});
|
|
}
|
|
|
|
function edge(point, size) {
|
|
var pad = [30, 100, 30, 100];
|
|
if (point[0] > size[0] - pad[0]) return [-10, 0];
|
|
else if (point[0] < pad[2]) return [10, 0];
|
|
else if (point[1] > size[1] - pad[1]) return [0, -10];
|
|
else if (point[1] < pad[3]) return [0, 10];
|
|
return null;
|
|
}
|
|
|
|
function startNudge(nudge) {
|
|
if (nudgeInterval) window.clearInterval(nudgeInterval);
|
|
nudgeInterval = window.setInterval(function() {
|
|
context.pan(nudge);
|
|
|
|
var orig = context.projection(origin);
|
|
|
|
orig = [orig[0] - nudge[1], orig[1] - nudge[1]];
|
|
delta = [delta[0] - nudge[0], delta[1] - nudge[1]];
|
|
origin = context.projection.invert(orig);
|
|
|
|
context.pop();
|
|
context.perform(
|
|
iD.actions.Move(entityIDs, delta, context.projection, cache),
|
|
annotation);
|
|
}, 50);
|
|
}
|
|
|
|
function stopNudge() {
|
|
if (nudgeInterval) window.clearInterval(nudgeInterval);
|
|
nudgeInterval = null;
|
|
}
|
|
|
|
function move() {
|
|
var mouse = context.mouse(),
|
|
orig = context.projection(origin);
|
|
|
|
delta = [mouse[0] - orig[0], mouse[1] - orig[1]];
|
|
|
|
var nudge = edge(mouse, context.map().dimensions());
|
|
if (nudge) startNudge(nudge);
|
|
else stopNudge();
|
|
|
|
context.pop();
|
|
context.perform(
|
|
iD.actions.Move(entityIDs, delta, context.projection, cache),
|
|
annotation);
|
|
}
|
|
|
|
function finish() {
|
|
d3.event.stopPropagation();
|
|
context.enter(iD.modes.Select(context, entityIDs)
|
|
.suppressMenu(true));
|
|
stopNudge();
|
|
}
|
|
|
|
function cancel() {
|
|
context.pop();
|
|
context.enter(iD.modes.Select(context, entityIDs)
|
|
.suppressMenu(true));
|
|
stopNudge();
|
|
}
|
|
|
|
function undone() {
|
|
context.enter(iD.modes.Browse(context));
|
|
}
|
|
|
|
mode.enter = function() {
|
|
clearCache();
|
|
cacheLocs(entityIDs);
|
|
origin = context.map().mouseCoordinates();
|
|
delta = [0, 0];
|
|
|
|
context.install(edit);
|
|
|
|
context.perform(
|
|
iD.actions.Noop(),
|
|
annotation);
|
|
|
|
context.surface()
|
|
.on('mousemove.move', move)
|
|
.on('click.move', finish);
|
|
|
|
context.history()
|
|
.on('undone.move', undone);
|
|
|
|
keybinding
|
|
.on('⎋', cancel)
|
|
.on('↩', finish);
|
|
|
|
d3.select(document)
|
|
.call(keybinding);
|
|
};
|
|
|
|
mode.exit = function() {
|
|
stopNudge();
|
|
|
|
context.uninstall(edit);
|
|
|
|
context.surface()
|
|
.on('mousemove.move', null)
|
|
.on('click.move', null);
|
|
|
|
context.history()
|
|
.on('undone.move', null);
|
|
|
|
keybinding.off();
|
|
};
|
|
|
|
return mode;
|
|
};
|