mirror of
https://github.com/FoggedLens/iD.git
synced 2026-02-13 17:23:02 +00:00
94 lines
2.9 KiB
JavaScript
94 lines
2.9 KiB
JavaScript
import _ from 'lodash';
|
|
// Join ways at the end node they share.
|
|
//
|
|
// This is the inverse of `iD.actions.Split`.
|
|
//
|
|
// Reference:
|
|
// https://github.com/systemed/potlatch2/blob/master/net/systemeD/halcyon/connection/actions/MergeWaysAction.as
|
|
// https://github.com/openstreetmap/josm/blob/mirror/src/org/openstreetmap/josm/actions/CombineWayAction.java
|
|
//
|
|
import { DeleteWay } from './delete_way';
|
|
import { interestingTag } from '../core/index';
|
|
import { joinWays } from '../geo/index';
|
|
|
|
export function Join(ids) {
|
|
|
|
function groupEntitiesByGeometry(graph) {
|
|
var entities = ids.map(function(id) { return graph.entity(id); });
|
|
return _.extend({line: []}, _.groupBy(entities, function(entity) { return entity.geometry(graph); }));
|
|
}
|
|
|
|
var action = function(graph) {
|
|
var ways = ids.map(graph.entity, graph),
|
|
survivor = ways[0];
|
|
|
|
// Prefer to keep an existing way.
|
|
for (var i = 0; i < ways.length; i++) {
|
|
if (!ways[i].isNew()) {
|
|
survivor = ways[i];
|
|
break;
|
|
}
|
|
}
|
|
|
|
var joined = joinWays(ways, graph)[0];
|
|
|
|
survivor = survivor.update({nodes: _.map(joined.nodes, 'id')});
|
|
graph = graph.replace(survivor);
|
|
|
|
joined.forEach(function(way) {
|
|
if (way.id === survivor.id)
|
|
return;
|
|
|
|
graph.parentRelations(way).forEach(function(parent) {
|
|
graph = graph.replace(parent.replaceMember(way, survivor));
|
|
});
|
|
|
|
survivor = survivor.mergeTags(way.tags);
|
|
|
|
graph = graph.replace(survivor);
|
|
graph = DeleteWay(way.id)(graph);
|
|
});
|
|
|
|
return graph;
|
|
};
|
|
|
|
action.disabled = function(graph) {
|
|
var geometries = groupEntitiesByGeometry(graph);
|
|
if (ids.length < 2 || ids.length !== geometries.line.length)
|
|
return 'not_eligible';
|
|
|
|
var joined = joinWays(ids.map(graph.entity, graph), graph);
|
|
if (joined.length > 1)
|
|
return 'not_adjacent';
|
|
|
|
var nodeIds = _.map(joined[0].nodes, 'id').slice(1, -1),
|
|
relation,
|
|
tags = {},
|
|
conflicting = false;
|
|
|
|
joined[0].forEach(function(way) {
|
|
var parents = graph.parentRelations(way);
|
|
parents.forEach(function(parent) {
|
|
if (parent.isRestriction() && parent.members.some(function(m) { return nodeIds.indexOf(m.id) >= 0; }))
|
|
relation = parent;
|
|
});
|
|
|
|
for (var k in way.tags) {
|
|
if (!(k in tags)) {
|
|
tags[k] = way.tags[k];
|
|
} else if (tags[k] && interestingTag(k) && tags[k] !== way.tags[k]) {
|
|
conflicting = true;
|
|
}
|
|
}
|
|
});
|
|
|
|
if (relation)
|
|
return 'restriction';
|
|
|
|
if (conflicting)
|
|
return 'conflicting_tags';
|
|
};
|
|
|
|
return action;
|
|
}
|