Files
iD/modules/operations/delete.js
Bryan Housel 966f5c3586 Add explicit validation calls to each operation
This is cleaner than adding hooks to transitionable actions, or performing validation
on every history change.  We want validation to run after each operation, but not
while the user is drawing lines or typing in fields.
2019-05-06 14:49:46 -04:00

155 lines
5.2 KiB
JavaScript

import { t } from '../util/locale';
import { actionDeleteMultiple } from '../actions/delete_multiple';
import { behaviorOperation } from '../behavior/operation';
import { geoExtent, geoSphericalDistance } from '../geo';
import { modeBrowse } from '../modes/browse';
import { modeSelect } from '../modes/select';
import { uiCmd } from '../ui/cmd';
import { utilGetAllNodes } from '../util';
export function operationDelete(selectedIDs, context) {
var multi = (selectedIDs.length === 1 ? 'single' : 'multiple');
var action = actionDeleteMultiple(selectedIDs);
var nodes = utilGetAllNodes(selectedIDs, context.graph());
var coords = nodes.map(function(n) { return n.loc; });
var extent = nodes.reduce(function(extent, node) {
return extent.extend(node.extent(context.graph()));
}, geoExtent());
var operation = function() {
var nextSelectedID;
var nextSelectedLoc;
if (selectedIDs.length === 1) {
var id = selectedIDs[0];
var entity = context.entity(id);
var geometry = context.geometry(id);
var parents = context.graph().parentWays(entity);
var parent = parents[0];
// Select the next closest node in the way.
if (geometry === 'vertex') {
var nodes = parent.nodes;
var i = nodes.indexOf(id);
if (i === 0) {
i++;
} else if (i === nodes.length - 1) {
i--;
} else {
var a = geoSphericalDistance(entity.loc, context.entity(nodes[i - 1]).loc);
var b = geoSphericalDistance(entity.loc, context.entity(nodes[i + 1]).loc);
i = a < b ? i - 1 : i + 1;
}
nextSelectedID = nodes[i];
nextSelectedLoc = context.entity(nextSelectedID).loc;
}
}
context.perform(action, operation.annotation());
context.validator().validate();
if (nextSelectedID && nextSelectedLoc) {
if (context.hasEntity(nextSelectedID)) {
context.enter(modeSelect(context, [nextSelectedID]).follow(true));
} else {
context.map().centerEase(nextSelectedLoc);
context.enter(modeBrowse(context));
}
} else {
context.enter(modeBrowse(context));
}
};
operation.available = function() {
return true;
};
operation.disabled = function() {
if (extent.area() && extent.percentContainedIn(context.extent()) < 0.8) {
return 'too_large';
} else if (someMissing()) {
return 'not_downloaded';
} else if (selectedIDs.some(context.hasHiddenConnections)) {
return 'connected_to_hidden';
} else if (selectedIDs.some(protectedMember)) {
return 'part_of_relation';
} else if (selectedIDs.some(incompleteRelation)) {
return 'incomplete_relation';
} else if (selectedIDs.some(hasWikidataTag)) {
return 'has_wikidata_tag';
}
return false;
function someMissing() {
var osm = context.connection();
if (osm) {
var missing = coords.filter(function(loc) { return !osm.isDataLoaded(loc); });
if (missing.length) {
missing.forEach(function(loc) { context.loadTileAtLoc(loc); });
return true;
}
}
return false;
}
function hasWikidataTag(id) {
var entity = context.entity(id);
return entity.tags.wikidata && entity.tags.wikidata.trim().length > 0;
}
function incompleteRelation(id) {
var entity = context.entity(id);
return entity.type === 'relation' && !entity.isComplete(context.graph());
}
function protectedMember(id) {
var entity = context.entity(id);
if (entity.type !== 'way') return false;
var parents = context.graph().parentRelations(entity);
for (var i = 0; i < parents.length; i++) {
var parent = parents[i];
var type = parent.tags.type;
var role = parent.memberById(id).role || 'outer';
if (type === 'route' || type === 'boundary' || (type === 'multipolygon' && role === 'outer')) {
return true;
}
}
return false;
}
};
operation.tooltip = function() {
var disable = operation.disabled();
return disable ?
t('operations.delete.' + disable + '.' + multi) :
t('operations.delete.description' + '.' + multi);
};
operation.annotation = function() {
return selectedIDs.length === 1 ?
t('operations.delete.annotation.' + context.geometry(selectedIDs[0])) :
t('operations.delete.annotation.multiple', { n: selectedIDs.length });
};
operation.id = 'delete';
operation.keys = [uiCmd('⌘⌫'), uiCmd('⌘⌦'), uiCmd('⌦')];
operation.title = t('operations.delete.title');
operation.behavior = behaviorOperation(context).which(operation);
return operation;
}