From 45b7a405579357ba18bcb3a8121d81b136ad755b Mon Sep 17 00:00:00 2001 From: Bryan Housel Date: Thu, 12 Jan 2017 19:30:07 +0530 Subject: [PATCH] Fix area drawing for #3676 These changes are needed now that `addNode` * wants to preserve circularity * automatically remove duplicates * range checks index argument (can't call it with -1 anymore) --- modules/behavior/draw_way.js | 74 ++++++++++++++++++++++-------------- modules/modes/add_area.js | 13 +++++-- modules/modes/draw_area.js | 11 +++--- 3 files changed, 61 insertions(+), 37 deletions(-) diff --git a/modules/behavior/draw_way.js b/modules/behavior/draw_way.js index db30811771..396b4a2a50 100644 --- a/modules/behavior/draw_way.js +++ b/modules/behavior/draw_way.js @@ -41,20 +41,24 @@ export function behaviorDrawWay(context, wayId, index, mode, baseGraph) { annotation = t((way.isDegenerate() ? 'operations.start.annotation.' : 'operations.continue.annotation.') + context.geometry(wayId)), - draw = behaviorDraw(context); + draw = behaviorDraw(context), + end = osmNode({ loc: context.map().mouseCoordinates() }), + startIndex, start, segment; - var startIndex = typeof index === 'undefined' ? way.nodes.length - 1 : 0, - start = osmNode({loc: context.graph().entity(way.nodes[startIndex]).loc}), - end = osmNode({loc: context.map().mouseCoordinates()}), + if (!isArea) { + startIndex = typeof index === 'undefined' ? way.nodes.length - 1 : 0; + start = osmNode({ loc: context.entity(way.nodes[startIndex]).loc }); segment = osmWay({ nodes: typeof index === 'undefined' ? [start.id, end.id] : [end.id, start.id], tags: _.clone(way.tags) }); + } + var fn = context[way.isDegenerate() ? 'replace' : 'perform']; if (isArea) { fn(actionAddEntity(end), - actionAddVertex(wayId, end.id, index) + actionAddVertex(wayId, end.id) ); } else { fn(actionAddEntity(start), @@ -70,7 +74,7 @@ export function behaviorDrawWay(context, wayId, index, mode, baseGraph) { if (datum.type === 'node' && datum.id !== end.id) { loc = datum.loc; - } else if (datum.type === 'way' && datum.id !== segment.id) { + } else if (datum.type === 'way') { // && (segment || datum.id !== segment.id)) { var dims = context.map().dimensions(), mouse = context.mouse(), pad = 5, @@ -145,9 +149,8 @@ export function behaviorDrawWay(context, wayId, index, mode, baseGraph) { return function(graph) { if (isArea) { return graph - .replace(way.addNode(newNode.id, index)) + .replace(way.addNode(newNode.id)) .remove(end); - } else { return graph .replace(graph.entity(wayId).addNode(newNode.id, index)) @@ -165,13 +168,19 @@ export function behaviorDrawWay(context, wayId, index, mode, baseGraph) { var last = context.hasEntity(way.nodes[way.nodes.length - (isArea ? 2 : 1)]); if (last && last.loc[0] === loc[0] && last.loc[1] === loc[1]) return; - var newNode = osmNode({loc: loc}); - - context.replace( - actionAddEntity(newNode), - ReplaceTemporaryNode(newNode), - annotation - ); + if (isArea) { + context.replace( + actionMoveNode(end.id, loc), + annotation + ); + } else { + var newNode = osmNode({loc: loc}); + context.replace( + actionAddEntity(newNode), + ReplaceTemporaryNode(newNode), + annotation + ); + } finished = true; context.enter(mode); @@ -180,21 +189,28 @@ export function behaviorDrawWay(context, wayId, index, mode, baseGraph) { // Connect the way to an existing way. drawWay.addWay = function(loc, edge) { - var previousEdge = startIndex ? - [way.nodes[startIndex], way.nodes[startIndex - 1]] : - [way.nodes[0], way.nodes[1]]; - // Avoid creating duplicate segments - if (!isArea && geoEdgeEqual(edge, previousEdge)) - return; - - var newNode = osmNode({ loc: loc }); - - context.perform( - actionAddMidpoint({ loc: loc, edge: edge}, newNode), - ReplaceTemporaryNode(newNode), - annotation - ); + if (isArea) { + context.perform( + actionAddMidpoint({ loc: loc, edge: edge}, end), + annotation + ); + } else { + var previousEdge = startIndex ? + [way.nodes[startIndex], way.nodes[startIndex - 1]] : + [way.nodes[0], way.nodes[1]]; + + // Avoid creating duplicate segments + if (geoEdgeEqual(edge, previousEdge)) + return; + + var newNode = osmNode({ loc: loc }); + context.perform( + actionAddMidpoint({ loc: loc, edge: edge}, newNode), + ReplaceTemporaryNode(newNode), + annotation + ); + } finished = true; context.enter(mode); diff --git a/modules/modes/add_area.js b/modules/modes/add_area.js index cf92eab680..f36ebef16a 100644 --- a/modules/modes/add_area.js +++ b/modules/modes/add_area.js @@ -27,6 +27,13 @@ export function modeAddArea(context) { defaultTags = { area: 'yes' }; + function actionClose(wayId) { + return function (graph) { + return graph.replace(graph.entity(wayId).close()); + }; + } + + function start(loc) { var graph = context.graph(), node = osmNode({ loc: loc }), @@ -36,7 +43,7 @@ export function modeAddArea(context) { actionAddEntity(node), actionAddEntity(way), actionAddVertex(way.id, node.id), - actionAddVertex(way.id, node.id) + actionClose(way.id) ); context.enter(modeDrawArea(context, way.id, graph)); @@ -52,7 +59,7 @@ export function modeAddArea(context) { actionAddEntity(node), actionAddEntity(way), actionAddVertex(way.id, node.id), - actionAddVertex(way.id, node.id), + actionClose(way.id), actionAddMidpoint({ loc: loc, edge: edge }, node) ); @@ -67,7 +74,7 @@ export function modeAddArea(context) { context.perform( actionAddEntity(way), actionAddVertex(way.id, node.id), - actionAddVertex(way.id, node.id) + actionClose(way.id) ); context.enter(modeDrawArea(context, way.id, graph)); diff --git a/modules/modes/draw_area.js b/modules/modes/draw_area.js index 763a459447..643ec9afe5 100644 --- a/modules/modes/draw_area.js +++ b/modules/modes/draw_area.js @@ -11,17 +11,18 @@ export function modeDrawArea(context, wayId, baseGraph) { mode.enter = function() { - var way = context.entity(wayId), - headId = way.nodes[way.nodes.length - 2], - tailId = way.first(); + var way = context.entity(wayId); - behavior = behaviorDrawWay(context, wayId, -1, mode, baseGraph) + behavior = behaviorDrawWay(context, wayId, undefined, mode, baseGraph) .tail(t('modes.draw_area.tail')); var addNode = behavior.addNode; behavior.addNode = function(node) { - if (node.id === headId || node.id === tailId) { + var length = way.nodes.length, + penultimate = length > 2 ? way.nodes[length - 2] : null; + + if (node.id === way.first() || node.id === penultimate) { behavior.finish(); } else { addNode(node);