modules/modes/drag_node.js (375 lines of code) (raw):
import { select as d3_select } from 'd3-selection';
import { vecSubtract, geomViewportNudge } from '@id-sdk/math';
import { utilArrayIntersection } from '@id-sdk/util';
import { actionAddMidpoint } from '../actions/add_midpoint';
import { actionConnect } from '../actions/connect';
import { actionMoveNode } from '../actions/move_node';
import { actionNoop } from '../actions/noop';
import { behaviorDrag } from '../behavior/drag';
import { behaviorEdit } from '../behavior/edit';
import { behaviorHover } from '../behavior/hover';
import { geoChooseEdge, geoHasLineIntersections, geoHasSelfIntersections } from '../geo';
import { modeBrowse } from './browse';
import { modeSelect } from './select';
import { osmJoinWays, osmNode } from '../osm';
import { presetManager } from '../presets';
import { t } from '../core/localizer';
import { utilKeybinding } from '../util';
export function modeDragNode(context) {
var mode = {
id: 'drag-node',
button: 'browse'
};
var hover = behaviorHover(context).altDisables(true)
.on('hover', context.ui().sidebar.hover);
var edit = behaviorEdit(context);
var _nudgeInterval;
var _restoreSelectedIDs = [];
var _wasMidpoint = false;
var _isCancelled = false;
var _activeEntity;
var _startLoc;
var _lastLoc;
function startNudge(d3_event, entity, nudge) {
if (_nudgeInterval) window.clearInterval(_nudgeInterval);
_nudgeInterval = window.setInterval(function() {
context.map().pan(nudge);
doMove(d3_event, entity, nudge);
}, 50);
}
function stopNudge() {
if (_nudgeInterval) {
window.clearInterval(_nudgeInterval);
_nudgeInterval = null;
}
}
function moveAnnotation(entity) {
return t('operations.move.annotation.' + entity.geometry(context.graph()));
}
function connectAnnotation(nodeEntity, targetEntity) {
var nodeGeometry = nodeEntity.geometry(context.graph());
var targetGeometry = targetEntity.geometry(context.graph());
if (nodeGeometry === 'vertex' && targetGeometry === 'vertex') {
var nodeParentWayIDs = context.graph().parentWays(nodeEntity);
var targetParentWayIDs = context.graph().parentWays(targetEntity);
var sharedParentWays = utilArrayIntersection(nodeParentWayIDs, targetParentWayIDs);
// if both vertices are part of the same way
if (sharedParentWays.length !== 0) {
// if the nodes are next to each other, they are merged
if (sharedParentWays[0].areAdjacent(nodeEntity.id, targetEntity.id)) {
return t('operations.connect.annotation.from_vertex.to_adjacent_vertex');
}
return t('operations.connect.annotation.from_vertex.to_sibling_vertex');
}
}
return t('operations.connect.annotation.from_' + nodeGeometry + '.to_' + targetGeometry);
}
function shouldSnapToNode(target) {
if (!_activeEntity) return false;
return _activeEntity.geometry(context.graph()) !== 'vertex' ||
(target.geometry(context.graph()) === 'vertex' || presetManager.allowsVertex(target, context.graph()));
}
function origin(entity) {
return context.projection(entity.loc);
}
function keydown(d3_event) {
if (d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
if (context.surface().classed('nope')) {
context.surface()
.classed('nope-suppressed', true);
}
context.surface()
.classed('nope', false)
.classed('nope-disabled', true);
}
}
function keyup(d3_event) {
if (d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
if (context.surface().classed('nope-suppressed')) {
context.surface()
.classed('nope', true);
}
context.surface()
.classed('nope-suppressed', false)
.classed('nope-disabled', false);
}
}
function start(d3_event, entity) {
_wasMidpoint = entity.type === 'midpoint';
var hasHidden = context.features().hasHiddenConnections(entity, context.graph());
_isCancelled = !context.editable() || d3_event.shiftKey || hasHidden;
if (_isCancelled) {
if (hasHidden) {
context.ui().flash
.duration(4000)
.iconName('#iD-icon-no')
.label(t('modes.drag_node.connected_to_hidden'))();
}
return drag.cancel();
}
if (_wasMidpoint) {
var midpoint = entity;
entity = osmNode();
context.perform(actionAddMidpoint(midpoint, entity));
entity = context.entity(entity.id); // get post-action entity
var vertex = context.surface().selectAll('.' + entity.id);
drag.targetNode(vertex.node())
.targetEntity(entity);
} else {
context.perform(actionNoop());
}
_activeEntity = entity;
_startLoc = entity.loc;
hover.ignoreVertex(entity.geometry(context.graph()) === 'vertex');
context.surface().selectAll('.' + _activeEntity.id)
.classed('active', true);
context.enter(mode);
}
// related code
// - `behavior/draw.js` `datum()`
function datum(d3_event) {
if (!d3_event || d3_event.altKey) {
return {};
} else {
// When dragging, snap only to touch targets..
// (this excludes area fills and active drawing elements)
var d = d3_event.target.__data__;
return (d && d.properties && d.properties.target) ? d : {};
}
}
function doMove(d3_event, entity, nudge) {
nudge = nudge || [0, 0];
var currPoint = (d3_event && d3_event.point) || context.projection(_lastLoc);
var currMouse = vecSubtract(currPoint, nudge);
var loc = context.projection.invert(currMouse);
var target, edge;
if (!_nudgeInterval) { // If not nudging at the edge of the viewport, try to snap..
// related code
// - `mode/drag_node.js` `doMove()`
// - `behavior/draw.js` `click()`
// - `behavior/draw_way.js` `move()`
var d = datum(d3_event);
target = d && d.properties && d.properties.entity;
var targetLoc = target && target.loc;
var targetNodes = d && d.properties && d.properties.nodes;
if (targetLoc) { // snap to node/vertex - a point target with `.loc`
if (shouldSnapToNode(target)) {
loc = targetLoc;
}
} else if (targetNodes) { // snap to way - a line target with `.nodes`
edge = geoChooseEdge(targetNodes, context.map().mouse(), context.projection, end.id);
if (edge) {
loc = edge.loc;
}
}
}
context.replace(
actionMoveNode(entity.id, loc)
);
// Below here: validations
var isInvalid = false;
// Check if this connection to `target` could cause relations to break..
if (target) {
isInvalid = hasRelationConflict(entity, target, edge, context.graph());
}
// Check if this drag causes the geometry to break..
if (!isInvalid) {
isInvalid = hasInvalidGeometry(entity, context.graph());
}
var nope = context.surface().classed('nope');
if (isInvalid === 'relation' || isInvalid === 'restriction') {
if (!nope) { // about to nope - show hint
context.ui().flash
.duration(4000)
.iconName('#iD-icon-no')
.label(t('operations.connect.' + isInvalid,
{ relation: presetManager.item('type/restriction').name() }
))();
}
} else if (isInvalid) {
var errorID = isInvalid === 'line' ? 'lines' : 'areas';
context.ui().flash
.duration(3000)
.iconName('#iD-icon-no')
.label(t('self_intersection.error.' + errorID))();
} else {
if (nope) { // about to un-nope, remove hint
context.ui().flash
.duration(1)
.label('')();
}
}
var nopeDisabled = context.surface().classed('nope-disabled');
if (nopeDisabled) {
context.surface()
.classed('nope', false)
.classed('nope-suppressed', isInvalid);
} else {
context.surface()
.classed('nope', isInvalid)
.classed('nope-suppressed', false);
}
_lastLoc = loc;
}
// Uses `actionConnect.disabled()` to know whether this connection is ok..
function hasRelationConflict(entity, target, edge, graph) {
var testGraph = graph.update(); // copy
// if snapping to way - add midpoint there and consider that the target..
if (edge) {
var midpoint = osmNode();
var action = actionAddMidpoint({
loc: edge.loc,
edge: [target.nodes[edge.index - 1], target.nodes[edge.index]]
}, midpoint);
testGraph = action(testGraph);
target = midpoint;
}
// can we connect to it?
var ids = [entity.id, target.id];
return actionConnect(ids).disabled(testGraph);
}
function hasInvalidGeometry(entity, graph) {
var parents = graph.parentWays(entity);
var i, j, k;
for (i = 0; i < parents.length; i++) {
var parent = parents[i];
var nodes = [];
var activeIndex = null; // which multipolygon ring contains node being dragged
// test any parent multipolygons for valid geometry
var relations = graph.parentRelations(parent);
for (j = 0; j < relations.length; j++) {
if (!relations[j].isMultipolygon()) continue;
var rings = osmJoinWays(relations[j].members, graph);
// find active ring and test it for self intersections
for (k = 0; k < rings.length; k++) {
nodes = rings[k].nodes;
if (nodes.find(function(n) { return n.id === entity.id; })) {
activeIndex = k;
if (geoHasSelfIntersections(nodes, entity.id)) {
return 'multipolygonMember';
}
}
rings[k].coords = nodes.map(function(n) { return n.loc; });
}
// test active ring for intersections with other rings in the multipolygon
for (k = 0; k < rings.length; k++) {
if (k === activeIndex) continue;
// make sure active ring doesn't cross passive rings
if (geoHasLineIntersections(rings[activeIndex].nodes, rings[k].nodes, entity.id)) {
return 'multipolygonRing';
}
}
}
// If we still haven't tested this node's parent way for self-intersections.
// (because it's not a member of a multipolygon), test it now.
if (activeIndex === null) {
nodes = parent.nodes.map(function(nodeID) { return graph.entity(nodeID); });
if (nodes.length && geoHasSelfIntersections(nodes, entity.id)) {
return parent.geometry(graph);
}
}
}
return false;
}
function move(d3_event, entity, point) {
if (_isCancelled) return;
d3_event.stopPropagation();
context.surface().classed('nope-disabled', d3_event.altKey);
_lastLoc = context.projection.invert(point);
doMove(d3_event, entity);
var nudge = geomViewportNudge(point, context.map().dimensions());
if (nudge) {
startNudge(d3_event, entity, nudge);
} else {
stopNudge();
}
}
function end(d3_event, entity) {
if (_isCancelled) return;
var wasPoint = entity.geometry(context.graph()) === 'point';
var d = datum(d3_event);
var nope = (d && d.properties && d.properties.nope) || context.surface().classed('nope');
var target = d && d.properties && d.properties.entity; // entity to snap to
if (nope) { // bounce back
context.perform(
_actionBounceBack(entity.id, _startLoc)
);
} else if (target && target.type === 'way') {
var choice = geoChooseEdge(context.graph().childNodes(target), context.map().mouse(), context.projection, entity.id);
context.replace(
actionAddMidpoint({
loc: choice.loc,
edge: [target.nodes[choice.index - 1], target.nodes[choice.index]]
}, entity),
connectAnnotation(entity, target)
);
} else if (target && target.type === 'node' && shouldSnapToNode(target)) {
context.replace(
actionConnect([target.id, entity.id]),
connectAnnotation(entity, target)
);
} else if (_wasMidpoint) {
context.replace(
actionNoop(),
t('operations.add.annotation.vertex')
);
} else {
context.replace(
actionNoop(),
moveAnnotation(entity)
);
}
if (wasPoint) {
context.enter(modeSelect(context, [entity.id]));
} else {
var reselection = _restoreSelectedIDs.filter(function(id) {
return context.graph().hasEntity(id);
});
if (reselection.length) {
context.enter(modeSelect(context, reselection));
} else {
context.enter(modeBrowse(context));
}
}
}
function _actionBounceBack(nodeID, toLoc) {
var moveNode = actionMoveNode(nodeID, toLoc);
var action = function(graph, t) {
// last time through, pop off the bounceback perform.
// it will then overwrite the initial perform with a moveNode that does nothing
if (t === 1) context.pop();
return moveNode(graph, t);
};
action.transitionable = true;
return action;
}
function cancel() {
drag.cancel();
context.enter(modeBrowse(context));
}
var drag = behaviorDrag()
.selector('.layer-touch.points .target')
.surface(context.container().select('.main-map').node())
.origin(origin)
.on('start', start)
.on('move', move)
.on('end', end);
mode.enter = function() {
context.install(hover);
context.install(edit);
d3_select(window)
.on('keydown.dragNode', keydown)
.on('keyup.dragNode', keyup);
context.history()
.on('undone.drag-node', cancel);
};
mode.exit = function() {
context.ui().sidebar.hover.cancel();
context.uninstall(hover);
context.uninstall(edit);
d3_select(window)
.on('keydown.dragNode', null)
.on('keyup.dragNode', null);
context.history()
.on('undone.drag-node', null);
_activeEntity = null;
context.surface()
.classed('nope', false)
.classed('nope-suppressed', false)
.classed('nope-disabled', false)
.selectAll('.active')
.classed('active', false);
stopNudge();
};
mode.selectedIDs = function() {
if (!arguments.length) return _activeEntity ? [_activeEntity.id] : [];
// no assign
return mode;
};
mode.activeID = function() {
if (!arguments.length) return _activeEntity && _activeEntity.id;
// no assign
return mode;
};
mode.restoreSelectedIDs = function(_) {
if (!arguments.length) return _restoreSelectedIDs;
_restoreSelectedIDs = _;
return mode;
};
mode.behavior = drag;
return mode;
}