]> git.openstreetmap.org Git - rails.git/blobdiff - vendor/assets/iD/iD.js
Update to iD v2.18.5
[rails.git] / vendor / assets / iD / iD.js
index ba3246b5f48ac1bb3616031fa6be2bbb6407686c..c3bc0c3eb9db668432873791dbf016cfce1bf2ec 100644 (file)
@@ -1,6 +1,10 @@
 (function () {
        var commonjsGlobal = typeof globalThis !== 'undefined' ? globalThis : typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : {};
 
+       function getDefaultExportFromCjs (x) {
+               return x && x.__esModule && Object.prototype.hasOwnProperty.call(x, 'default') ? x['default'] : x;
+       }
+
        function createCommonjsModule(fn, basedir, module) {
                return module = {
                  path: basedir,
                }, fn(module, module.exports), module.exports;
        }
 
-       function getCjsExportFromNamespace (n) {
-               return n && n['default'] || n;
-       }
-
        function commonjsRequire () {
                throw new Error('Dynamic requires are not currently supported by @rollup/plugin-commonjs');
        }
                if (_loadPromise) { return _loadPromise; }
 
                return _loadPromise = Promise.all([
-                       // load the list of langauges
+                       // load the list of languages
                        _mainFileFetcher.get('languages'),
                        // load the list of supported locales
                        _mainFileFetcher.get('locales')
 
            localizer.languageName = function (code, options) {
 
-               if (_languageNames[code]) {  // name in locale langauge
+               if (_languageNames[code]) {  // name in locale language
                  // e.g. "German"
                  return _languageNames[code];
                }
                    return localizer.t('translate.language_and_code', { language: langInfo.nativeName, code: code });
 
                  } else if (langInfo.base && langInfo.script) {
-                   var base = langInfo.base;   // the code of the langauge this is based on
+                   var base = langInfo.base;   // the code of the language this is based on
 
-                   if (_languageNames[base]) {   // base language name in locale langauge
+                   if (_languageNames[base]) {   // base language name in locale language
                      var scriptCode = langInfo.script;
                      var script = _scriptNames[scriptCode] || scriptCode;
                      // e.g. "Serbian (Cyrillic)"
            if (_this.suggestion) {
              var path = presetID.split('/');
              path.pop();  // remove brand name
-             // NOTE: insert an en-dash, not a hypen (to avoid conflict with fr - nl names in Brussels etc)
+             // NOTE: insert an en-dash, not a hyphen (to avoid conflict with fr - nl names in Brussels etc)
              return _this.originalName + ' – ' + _t('presets.presets.' + path.join('/') + '.name');
            }
            return _this.t('name', { 'default': _this.originalName });
          };
 
 
-         _this.reference = function (geom) {
+         _this.reference = function () {
            // Lookup documentation on Wikidata...
            var qid = _this.tags.wikidata || _this.tags['brand:wikidata'] || _this.tags['operator:wikidata'];
            if (qid) {
            var key = _this.originalReference.key || Object.keys(utilObjectOmit(_this.tags, 'name'))[0];
            var value = _this.originalReference.value || _this.tags[key];
 
-           if (geom === 'relation' && key === 'type') {
-             if (value in _this.tags) {
-               key = value;
-               value = _this.tags[key];
-             } else {
-               return { rtype: value };
-             }
-           }
-
            if (value === '*') {
              return { key: key };
            } else {
                var sign = d3_polygonArea(points) > 0 ? 1 : -1;
                var ids, i, j, k;
 
-               // we need atleast two key nodes for the algorithm to work
+               // we need at least two key nodes for the algorithm to work
                if (!keyNodes.length) {
                    keyNodes = [nodes[0]];
                    keyPoints = [points[0]];
                }
 
                // key points and nodes are those connected to the ways,
-               // they are projected onto the circle, inbetween nodes are moved
-               // to constant intervals between key nodes, extra inbetween nodes are
+               // they are projected onto the circle, in between nodes are moved
+               // to constant intervals between key nodes, extra in between nodes are
                // added if necessary.
                for (i = 0; i < keyPoints.length; i++) {
                    var nextKeyNodeIndex = (i + 1) % keyNodes.length;
                        graph = graph.replace(node);
                    }
 
-                   // add new inbetween nodes if necessary
+                   // add new in between nodes if necessary
                    for (j = 0; j < numberNewPoints; j++) {
                        angle = startAngle + (indexRange + j) * eachAngle;
                        loc = projection.invert([
 
                    // Check for other ways that share these keyNodes..
                    // If keyNodes are adjacent in both ways,
-                   // we can add inBetween nodes to that shared way too..
+                   // we can add inBetweenNodes to that shared way too..
                    if (indexRange === 1 && inBetweenNodes.length) {
                        var startIndex1 = way.nodes.lastIndexOf(startNode.id);
                        var endIndex1 = way.nodes.lastIndexOf(endNode.id);
 
            var action = function(graph) {
                ids.forEach(function(id) {
-                   if (graph.hasEntity(id)) { // It may have been deleted aready.
+                   if (graph.hasEntity(id)) { // It may have been deleted already.
                        graph = actions[graph.entity(id).type](id)(graph);
                    }
                });
          };
        }
 
-       // Disconect the ways at the given node.
+       // Disconnect the ways at the given node.
        //
        // Optionally, disconnect only the given ways.
        //
                    extractedLoc = entity.extent(graph).center();
                }
 
-               var isBuilding = entity.tags.building && entity.tags.building !== 'no';
+               var indoorAreaValues = {
+                   area: true,
+                   corridor: true,
+                   elevator: true,
+                   level: true,
+                   room: true
+               };
+
+               var isBuilding = (entity.tags.building && entity.tags.building !== 'no') ||
+                   (entity.tags['building:part'] && entity.tags['building:part'] !== 'no');
+
+               var isIndoorArea = fromGeometry === 'area' && entity.tags.indoor && indoorAreaValues[entity.tags.indoor];
 
                var entityTags = Object.assign({}, entity.tags);  // shallow copy
                var pointTags = {};
                            key.match(/^building:.{1,}/) ||
                            key.match(/^roof:.{1,}/)) { continue; }
                    }
+                   // leave `indoor` tag on the area
+                   if (isIndoorArea && key === 'indoor') {
+                       continue;
+                   }
 
                    // copy the tag from the entity to the point
                    pointTags[key] = entityTags[key];
                    if (keysToCopyAndRetain.indexOf(key) !== -1 ||
                        key.match(/^addr:.{1,}/)) {
                        continue;
+                   } else if (isIndoorArea && key === 'level') {
+                       // leave `level` on both features
+                       continue;
                    }
 
                    // remove the tag from the entity
                    delete entityTags[key];
                }
 
-               if (!isBuilding && fromGeometry === 'area') {
+               if (!isBuilding && !isIndoorArea && fromGeometry === 'area') {
                    // ensure that areas keep area geometry
                    entityTags.area = 'yes';
                }
        QAItem.prototype.update = function update (props) {
            var this$1 = this;
 
-         // You can't override this inital information
+         // You can't override this initial information
          var ref = this;
            var loc = ref.loc;
            var service = ref.service;
                // prevent operations during low zoom selection
                if (!context.map().withinEditableZoom()) { return; }
 
+               if (_operation.availableForKeypress && !_operation.availableForKeypress()) { return; }
+
                event.preventDefault();
+
                var disabled = _operation.disabled();
 
                if (disabled) {
 
 
            function behavior(selection) {
-               _pointerId = null;
                var matchesSelector = utilPrefixDOMProperty('matchesSelector');
                var delegate = pointerdown;
 
                        for (k = 0; k < rings.length; k++) {
                            if (k === activeIndex) { continue; }
 
-                           // make sure active ring doesnt cross passive rings
+                           // make sure active ring doesn't cross passive rings
                            if (geoHasLineIntersections(rings[activeIndex].nodes, rings[k].nodes, entity.id)) {
                                return 'multipolygonRing';
                            }
          return _t(("QA.improveOSM.directions." + (compass[dir])));
        }
 
-       // Errors shouldn't obscure eachother
+       // Errors shouldn't obscure each other
        function preventCoincident(loc, bumpUp) {
          var coincident = false;
          do {
          }
        }
 
-       // Issues shouldn't obscure eachother
+       // Issues shouldn't obscure each other
        function preventCoincident$1(loc) {
          var coincident = false;
          do {
                //
                // There is some logic here to batch up clicks into a _mlyClicks array
                // because the user might click on a lot of markers quickly and nodechanged
-               // may be called out of order asychronously.
+               // may be called out of order asynchronously.
                //
                // Clicks are added to the array in `selectedImage` and removed here.
                //
                                return x.join('=');
                            }).join(','),
                        popup = window.open('about:blank', 'oauth_window', settings);
+
+                   oauth.popupWindow = popup;
+
+                   if (!popup) {
+                       var error = new Error('Popup was blocked');
+                       error.status = 'popup-blocked';
+                       throw error;
+                   }
                }
 
                // Request a request token. When this is complete, the popup
                }
            };
 
+           oauth.bringPopupWindowToFront = function() {
+               var brougtPopupToFront = false;
+               try {
+                   // This may cause a cross-origin error:
+                   // `DOMException: Blocked a frame with origin "..." from accessing a cross-origin frame.`
+                   if (oauth.popupWindow && !oauth.popupWindow.closed) {
+                       oauth.popupWindow.focus();
+                       brougtPopupToFront = true;
+                   }
+               } catch (err) {
+                   // Bringing popup window to front failed (probably because of the cross-origin error mentioned above)
+               }
+               return brougtPopupToFront;
+           };
+
            oauth.bootstrapToken = function(oauth_token, callback) {
                // ## Getting an request token
                // At this point we have an `oauth_token`, brought in from a function
            loading: authLoading,
            done: authDone
        });
-
-       var _blacklists = ['.*\.google(apis)?\..*/(vt|kh)[\?/].*([xyz]=.*){3}.*'];
+       // hardcode default block of Google Maps
+       var _imageryBlocklists = [/.*\.google(apis)?\..*\/(vt|kh)[\?\/].*([xyz]=.*){3}.*/];
        var _tileCache = { toLoad: {}, loaded: {}, inflight: {}, seen: {}, rtree: new RBush() };
        var _noteCache = { toLoad: {}, loaded: {}, inflight: {}, inflightPost: {}, note: {}, closed: {}, rtree: new RBush() };
        var _userCache = { toLoad: {}, user: {} };
                        return callback(err, null);
                    }
 
-                   // update blacklists
+                   // update blocklists
                    var elements = xml.getElementsByTagName('blacklist');
                    var regexes = [];
                    for (var i = 0; i < elements.length; i++) {
-                       var regex = elements[i].getAttribute('regex');  // needs unencode?
-                       if (regex) {
-                           regexes.push(regex);
+                       var regexString = elements[i].getAttribute('regex');  // needs unencode?
+                       if (regexString) {
+                           try {
+                               var regex = new RegExp(regexString);
+                               regexes.push(regex);
+                           } catch (e) {
+                               /* noop */
+                           }
                        }
                    }
                    if (regexes.length) {
-                       _blacklists = regexes;
+                       _imageryBlocklists = regexes;
                    }
 
                    if (_rateLimitError) {
            // Calls `status` and dispatches an `apiStatusChange` event if the returned
            // status differs from the cached status.
            reloadApiStatus: function() {
-               // throttle to avoid unncessary API calls
+               // throttle to avoid unnecessary API calls
                if (!this.throttledReloadApiStatus) {
                    var that = this;
                    this.throttledReloadApiStatus = throttle(function() {
            },
 
 
-           imageryBlacklists: function() {
-               return _blacklists;
+           imageryBlocklists: function() {
+               return _imageryBlocklists;
            },
 
 
            // {
            //   key: 'string',
            //   value: 'string',
-           //   rtype: 'string',
            //   langCode: 'string'
            // }
            //
                var that = this;
                var titles = [];
                var result = {};
-               var rtypeSitelink = params.rtype ? ('Relation:' + params.rtype).replace(/_/g, ' ').trim() : false;
+               var rtypeSitelink = (params.key === 'type' && params.value) ? ('Relation:' + params.value).replace(/_/g, ' ').trim() : false;
                var keySitelink = params.key ? this.toSitelink(params.key) : false;
                var tagSitelink = (params.key && params.value) ? this.toSitelink(params.key, params.value) : false;
                var localeSitelink;
            //   key: 'string',     // required
            //   value: 'string'    // optional
            // }
-           //   -or-
-           // {
-           //   rtype: 'rtype'     // relation type  (e.g. 'multipolygon')
-           // }
            //
            // Get an result object used to display tag documentation
            // {
        }
        });
 
+       var turf_bboxClip = /*@__PURE__*/getDefaultExportFromCjs(bboxClip_1);
+
        var fastJsonStableStringify = function (data, opts) {
            if (!opts) { opts = {}; }
            if (typeof opts === 'function') { opts = { cmp: opts }; }
                        // Clip to tile bounds
                        if (geometry.type === 'MultiPolygon') {
                            var isClipped = false;
-                           var featureClip = bboxClip_1(feature, tile.extent.rectangle());
+                           var featureClip = turf_bboxClip(feature, tile.extent.rectangle());
                            if (!fastDeepEqual(feature.geometry, featureClip.geometry)) {
                                // feature = featureClip;
                                isClipped = true;
                    type: 'item',
                    // the language to search
                    language: lang,
-                   // the langauge for the label and description in the result
+                   // the language for the label and description in the result
                    uselang: lang,
                    limit: 10,
                    origin: '*'
                mainEnter
                    .append('div')
                    .attr('class', 'comment-text')
-                   .html(function(d) { return d.html; });
+                   .html(function(d) { return d.html; })
+                   .selectAll('a')
+                       .attr('rel', 'noopener nofollow')
+                       .attr('target', '_blank');
 
                comments
                    .call(replaceAvatars);
            return section;
        }
 
-       // Pass `which` object of the form:
+       // Pass `what` object of the form:
        // {
        //   key: 'string',     // required
        //   value: 'string'    // optional
        // }
        //   -or-
        // {
-       //   rtype: 'string'    // relation type  (e.g. 'multipolygon')
-       // }
-       //   -or-
-       // {
        //   qid: 'string'      // brand wikidata  (e.g. 'Q37158')
        // }
        //
 
 
            tagReference.body = function(selection) {
-               var itemID = what.qid || what.rtype || (what.key + '-' + what.value);
+               var itemID = what.qid || (what.key + '-' + (what.value || ''));
                _body = selection.selectAll('.tag-reference-body')
                    .data([itemID], function(d) { return d; });
 
                            bindTypeahead(key, value);
                        }
 
-                       var reference;
-
-                       if (typeof d.value !== 'string') {
-                           reference = uiTagReference({ key: d.key });
-                       } else {
-                           var isRelation = _entityIDs && _entityIDs.some(function(entityID) {
-                               return context.entity(entityID).type === 'relation';
-                           });
-                           if (isRelation && d.key === 'type') {
-                               reference = uiTagReference({ rtype: d.value });
-                           } else {
-                               reference = uiTagReference({ key: d.key, value: d.value });
-                           }
+                       var referenceOptions = { key: d.key };
+                       if (typeof d.value === 'string') {
+                           referenceOptions.value = d.value;
                        }
+                       var reference = uiTagReference(referenceOptions);
 
                        if (_state === 'hover') {
                            reference.showing(false);
              .append('div')
                .attr('class', 'qa-details-subsection');
 
-           // Suggested Fix (musn't exist for every issue type)
+           // Suggested Fix (mustn't exist for every issue type)
            if (issueString(_qaItem, 'fix')) {
              var div$1 = detailsEnter
                .append('div')
                  .attr('target', '_blank');
            }
 
-           // Common Pitfalls (musn't exist for every issue type)
+           // Common Pitfalls (mustn't exist for every issue type)
            if (issueString(_qaItem, 'trap')) {
              var div$2 = detailsEnter
                .append('div')
                if (includeDrawNode) {
                    if (parentWay.isClosed()) {
                        // don't test the last segment for closed ways - #4655
-                       // (still test the first segement)
+                       // (still test the first segment)
                        nodes.pop();
                    }
                } else { // discount the draw node
 
                var nextMode;
 
-               if (context.graph() === startGraph) { // we've undone back to the beginning
+               if (context.graph() === startGraph) {
+                   // We've undone back to the initial state before we started drawing.
+                   // Just exit the draw mode without undoing whatever we did before
+                   // we entered the draw mode.
                    nextMode = modeSelect(context, [wayID]);
                } else {
-                   context.history()
-                       .on('undone.draw', null);
-                   // remove whatever segment was drawn previously
-                   context.undo();
+                   // The `undo` only removed the temporary edit, so here we have to
+                   // manually undo to actually remove the last node we added. We can't
+                   // use the `undo` function since the initial "add" graph doesn't have
+                   // an annotation and so cannot be undone to.
+                   context.pop(1);
 
-                   if (context.graph() === startGraph) { // we've undone back to the beginning
-                       nextMode = modeSelect(context, [wayID]);
-                   } else {
-                       // continue drawing
-                       nextMode = mode;
-                   }
+                   // continue drawing
+                   nextMode = mode;
                }
 
-               // clear the redo stack by adding and removing an edit
+               // clear the redo stack by adding and removing a blank edit
                context.perform(actionNoop());
                context.pop(1);
 
                _headNodeID = typeof _nodeIndex === 'number' ? _origWay.nodes[_nodeIndex] :
                    (_origWay.isClosed() ? _origWay.nodes[_origWay.nodes.length - 2] : _origWay.nodes[_origWay.nodes.length - 1]);
                _wayGeometry = _origWay.geometry(context.graph());
-               _annotation = _t((_origWay.isDegenerate() ?
+               _annotation = _t((_origWay.nodes.length === (_origWay.isClosed() ? 2 : 1) ?
                    'operations.start.annotation.' :
                    'operations.continue.annotation.') + _wayGeometry
                );
 
            var operation = function() {
 
-               if (!getSelectionText()) {
-                   event.preventDefault();
-               }
-
                var graph = context.graph();
                var selected = groupEntities(getFilteredIdsToCopy(), graph);
                var canCopy = [];
            }
 
 
-           function getSelectionText() {
-               return window.getSelection().toString();
-           }
-
-
            operation.available = function() {
                return getFilteredIdsToCopy().length > 0;
            };
            };
 
 
+           operation.availableForKeypress = function() {
+               var selection = window.getSelection && window.getSelection();
+               // if the user has text selected then let them copy that, not the selected feature
+               return !selection || !selection.toString();
+           };
+
+
            operation.tooltip = function() {
                var disable = operation.disabled();
                return disable ?
                });
                _nodes = utilGetAllNodes(_wayIDs, context.graph());
                _coords = _nodes.map(function(n) { return n.loc; });
-               _extent = utilTotalExtent(ways, context.graph());
 
                // actions for connected nodes shared by at least two selected ways
                var sharedActions = [];
+               var sharedNodes = [];
                // actions for connected nodes
                var unsharedActions = [];
+               var unsharedNodes = [];
 
                _nodes.forEach(function(node) {
                    var action = actionDisconnect(node.id).limitWays(_wayIDs);
 
                        if (count > 1) {
                            sharedActions.push(action);
+                           sharedNodes.push(node);
                        } else {
                            unsharedActions.push(action);
+                           unsharedNodes.push(node);
                        }
                    }
                });
                if (sharedActions.length) {
                    // if any nodes are shared, only disconnect the selected ways from each other
                    _actions = sharedActions;
+                   _extent = utilTotalExtent(sharedNodes, context.graph());
                    _descriptionID += 'conjoined';
                    _annotationID = 'from_each_other';
                } else {
                    // if no nodes are shared, disconnect the selected ways from all connected ways
                    _actions = unsharedActions;
+                   _extent = utilTotalExtent(unsharedNodes, context.graph());
                    if (_wayIDs.length === 1) {
                        _descriptionID += context.graph().geometry(_wayIDs[0]);
                    } else {
                for (var i = 0; i < selectedIDs.length; i++) {
                    var entity = context.hasEntity(selectedIDs[i]);
                    if (!entity || entity.geometry(graph) !== 'vertex') {
-                       return [];  // selection includes some not vertexes
+                       return [];  // selection includes some not vertices
                    }
 
                    var currParents = graph.parentWays(entity).map(function(w) { return w.id; });
                    var maxdist = 500;
 
                    // Don't allow the hash location to change too much while drawing
-                   // This can happen if the user accidently hit the back button.  #3996
+                   // This can happen if the user accidentally hit the back button.  #3996
                    if (mode && mode.id.match(/^draw/) !== null && dist > maxdist) {
                        context.enter(modeBrowse(context));
                        return;
 
            function segmentBBox(segment) {
                var extent = segment.extent(head);
-               // extent can be null if the node entites aren't in the graph for some reason
+               // extent can be null if the node entities aren't in the graph for some reason
                if (!extent) { return null; }
 
                var bbox = extent.bbox();
                return featureType === 'highway' || featureType === 'railway' || featureType === 'waterway';
            }
 
-
-           function getFeatureTypeForCrossingCheck(way, graph) {
-               var feature = getFeatureWithFeatureTypeTagsForWay(way, graph);
-               return getFeatureType(feature, graph);
-           }
-
            // discard
            var ignoredBuildings = {
                demolished: true, dismantled: true, proposed: true, razed: true
            }
 
 
-           function isLegitCrossing(way1, featureType1, way2, featureType2) {
-               var tags1 = way1.tags;
-               var tags2 = way2.tags;
+           function isLegitCrossing(tags1, featureType1, tags2, featureType2) {
 
                // assume 0 by default
                var level1 = tags1.level || '0';
                    var featureTypes = [featureType1, featureType2];
                    if (featureTypes.indexOf('highway') !== -1) {
                        if (featureTypes.indexOf('railway') !== -1) {
+                           if (!bothLines) { return {}; }
+
+                           var isTram = entity1.tags.railway === 'tram' || entity2.tags.railway === 'tram';
+
                            if (osmPathHighwayTagValues[entity1.tags.highway] ||
                                osmPathHighwayTagValues[entity2.tags.highway]) {
-                               // path-rail connections use this tag
-                               return bothLines ? { railway: 'crossing' } : {};
+
+                               // path-tram connections use this tag
+                               if (isTram) { return { railway: 'tram_crossing' }; }
+
+                               // other path-rail connections use this tag
+                               return { railway: 'crossing' };
                            } else {
-                               // road-rail connections use this tag
-                               return bothLines ? { railway: 'level_crossing' } : {};
+                               // path-tram connections use this tag
+                               if (isTram) { return { railway: 'tram_level_crossing' }; }
+
+                               // other road-rail connections use this tag
+                               return { railway: 'level_crossing' };
                            }
                        }
 
                var edgeCrossInfos = [];
                if (way1.type !== 'way') { return edgeCrossInfos; }
 
-               var way1FeatureType = getFeatureTypeForCrossingCheck(way1, graph);
+               var taggedFeature1 = getFeatureWithFeatureTypeTagsForWay(way1, graph);
+               var way1FeatureType = getFeatureType(taggedFeature1, graph);
                if (way1FeatureType === null) { return edgeCrossInfos; }
 
                var checkedSingleCrossingWays = {};
                var n1, n2, nA, nB, nAId, nBId;
                var segment1, segment2;
                var oneOnly;
-               var segmentInfos, segment2Info, way2, way2FeatureType;
+               var segmentInfos, segment2Info, way2, taggedFeature2, way2FeatureType;
                var way1Nodes = graph.childNodes(way1);
                var comparedWays = {};
                for (i = 0; i < way1Nodes.length - 1; i++) {
 
                        way2 = graph.hasEntity(segment2Info.wayId);
                        if (!way2) { continue; }
-
+                       taggedFeature2 = getFeatureWithFeatureTypeTagsForWay(way2, graph);
                        // only check crossing highway, waterway, building, and railway
-                       way2FeatureType = getFeatureTypeForCrossingCheck(way2, graph);
+                       way2FeatureType = getFeatureType(taggedFeature2, graph);
+
                        if (way2FeatureType === null ||
-                           isLegitCrossing(way1, way1FeatureType, way2, way2FeatureType)) {
+                           isLegitCrossing(taggedFeature1.tags, way1FeatureType, taggedFeature2.tags, way2FeatureType)) {
                            continue;
                        }
 
                        } else if (context.graph().geometry(this.entityIds[0]) === 'line' &&
                            context.graph().geometry(this.entityIds[1]) === 'line') {
 
-                           // don't recommend adding bridges to waterways since they're uncommmon
+                           // don't recommend adding bridges to waterways since they're uncommon
                            if (allowsBridge(selectedFeatureType) && selectedFeatureType !== 'waterway') {
                                fixes.push(makeAddBridgeOrTunnelFix('add_a_bridge', 'temaki-bridge', 'bridge'));
                            }
 
-                           // don't recommend adding tunnels under waterways since they're uncommmon
+                           // don't recommend adding tunnels under waterways since they're uncommon
                            var skipTunnelFix = otherFeatureType === 'waterway' && selectedFeatureType !== 'waterway';
                            if (allowsTunnel(selectedFeatureType) && !skipTunnelFix) {
                                fixes.push(makeAddBridgeOrTunnelFix('add_a_tunnel', 'temaki-tunnel', 'tunnel'));
        };
 
        var matchGroups = {adult_gaming_centre:["amenity/casino","amenity/gambling","leisure/adult_gaming_centre"],beauty:["shop/beauty","shop/hairdresser_supply"],bed:["shop/bed","shop/furniture"],beverages:["shop/alcohol","shop/beverages"],camping:["leisure/park","tourism/camp_site","tourism/caravan_site"],car_parts:["shop/car_parts","shop/car_repair","shop/tires","shop/tyres"],confectionery:["shop/candy","shop/chocolate","shop/confectionery"],convenience:["shop/beauty","shop/chemist","shop/convenience","shop/cosmetics","shop/newsagent"],coworking:["amenity/coworking_space","office/coworking","office/coworking_space"],electronics:["office/telecommunication","shop/computer","shop/electronics","shop/hifi","shop/mobile","shop/mobile_phone","shop/telecommunication"],fashion:["shop/accessories","shop/bag","shop/botique","shop/clothes","shop/department_store","shop/fashion","shop/fashion_accessories","shop/sports","shop/shoes"],financial:["amenity/bank","office/accountant","office/financial","office/financial_advisor","office/tax_advisor","shop/tax"],fitness:["leisure/fitness_centre","leisure/fitness_center","leisure/sports_centre","leisure/sports_center"],food:["amenity/cafe","amenity/fast_food","amenity/ice_cream","amenity/restaurant","shop/bakery","shop/ice_cream","shop/pastry","shop/tea","shop/coffee"],fuel:["amenity/fuel","shop/gas","shop/convenience;gas","shop/gas;convenience"],gift:["shop/gift","shop/card","shop/cards","shop/stationery"],hardware:["shop/carpet","shop/diy","shop/doityourself","shop/doors","shop/electrical","shop/flooring","shop/hardware","shop/power_tools","shop/tool_hire","shop/tools","shop/trade"],health_food:["shop/health","shop/health_food","shop/herbalist","shop/nutrition_supplements"],houseware:["shop/houseware","shop/interior_decoration"],lodging:["tourism/hotel","tourism/motel"],money_transfer:["amenity/money_transfer","shop/money_transfer"],outdoor:["shop/outdoor","shop/sports"],rental:["amenity/bicycle_rental","amenity/boat_rental","amenity/car_rental","amenity/truck_rental","amenity/vehicle_rental","shop/rental"],school:["amenity/childcare","amenity/college","amenity/kindergarten","amenity/language_school","amenity/prep_school","amenity/school","amenity/university"],supermarket:["shop/food","shop/frozen_food","shop/greengrocer","shop/grocery","shop/supermarket","shop/wholesale"],variety_store:["shop/variety_store","shop/supermarket","shop/discount","shop/convenience"],vending:["amenity/vending_machine","shop/vending_machine"],wholesale:["shop/wholesale","shop/supermarket","shop/department_store"]};
-       var match_groups = {
+       var require$$0 = {
        matchGroups: matchGroups
        };
 
-       var match_groups$1 = /*#__PURE__*/Object.freeze({
-               __proto__: null,
-               matchGroups: matchGroups,
-               'default': match_groups
-       });
-
-       var require$$0 = getCjsExportFromNamespace(match_groups$1);
-
        var matchGroups$1 = require$$0.matchGroups;
 
 
                .on('redone.validator', validator.validate);   // redo
                // but not on 'change' (e.g. while drawing)
 
-           // When user chages editing modes:
+           // When user changes editing modes:
            context
                .on('exit.validator', validator.validate);
 
            context.history().photoOverlaysUsed(photoOverlaysUsed);
          };
 
+         var _checkedBlocklists;
 
          background.sources = function (extent, zoom, includeCurrent) {
            if (!_imageryIndex) { return []; }   // called before init()?
 
            var currSource = baseLayer.source();
 
+           var osm = context.connection();
+           var blocklists = osm && osm.imageryBlocklists();
+
+           if (blocklists && blocklists !== _checkedBlocklists) {
+             _imageryIndex.backgrounds.forEach(function (source) {
+               source.isBlocked = blocklists.some(function(blocklist) {
+                 return blocklist.test(source.template());
+               });
+             });
+             _checkedBlocklists = blocklists;
+           }
+
            return _imageryIndex.backgrounds.filter(function (source) {
+             if (includeCurrent && currSource === source) { return true; }  // optionally always include the current imagery
+             if (source.isBlocked) { return false; }                        // even bundled sources may be blocked - #7905
              if (!source.polygon) { return true; }                          // always include imagery with worldwide coverage
-             if (includeCurrent && currSource === source) { return true; }  // optionally include the current imagery
              if (zoom && zoom < 6) { return false; }                        // optionally exclude local imagery at low zooms
              return visible[source.id];                                 // include imagery visible in given extent
            });
          background.baseLayerSource = function(d) {
            if (!arguments.length) { return baseLayer.source(); }
 
-           // test source against OSM imagery blacklists..
+           // test source against OSM imagery blocklists..
            var osm = context.connection();
            if (!osm) { return background; }
 
-           var blacklists = osm.imageryBlacklists();
+           var blocklists = osm.imageryBlocklists();
            var template = d.template();
            var fail = false;
            var tested = 0;
            var regex;
 
-           for (var i = 0; i < blacklists.length; i++) {
-             try {
-               regex = new RegExp(blacklists[i]);
-               fail = regex.test(template);
-               tested++;
-               if (fail) { break; }
-             } catch (e) {
-               /* noop */
-             }
+           for (var i = 0; i < blocklists.length; i++) {
+             regex = blocklists[i];
+             fail = regex.test(template);
+             tested++;
+             if (fail) { break; }
            }
 
            // ensure at least one test was run.
            if (!tested) {
-             regex = new RegExp('.*\.google(apis)?\..*/(vt|kh)[\?/].*([xyz]=.*){3}.*');
+             regex = /.*\.google(apis)?\..*\/(vt|kh)[\?\/].*([xyz]=.*){3}.*/;
              fail = regex.test(template);
            }
 
            drawData.template = function(val, src) {
                if (!arguments.length) { return _template; }
 
-               // test source against OSM imagery blacklists..
+               // test source against OSM imagery blocklists..
                var osm = context.connection();
                if (osm) {
-                   var blacklists = osm.imageryBlacklists();
+                   var blocklists = osm.imageryBlocklists();
                    var fail = false;
                    var tested = 0;
                    var regex;
 
-                   for (var i = 0; i < blacklists.length; i++) {
-                       try {
-                           regex = new RegExp(blacklists[i]);
-                           fail = regex.test(val);
-                           tested++;
-                           if (fail) { break; }
-                       } catch (e) {
-                           /* noop */
-                       }
+                   for (var i = 0; i < blocklists.length; i++) {
+                       regex = blocklists[i];
+                       fail = regex.test(val);
+                       tested++;
+                       if (fail) { break; }
                    }
 
                    // ensure at least one test was run.
                    if (!tested) {
-                       regex = new RegExp('.*\.google(apis)?\..*/(vt|kh)[\?/].*([xyz]=.*){3}.*');
+                       regex = /.*\.google(apis)?\..*\/(vt|kh)[\?\/].*([xyz]=.*){3}.*/;
                        fail = regex.test(val);
                    }
                }
 
            /**
             * drawImages()
-            * drawImages is the method that is returned (and that runs) everytime 'svgStreetside()' is called.
+            * drawImages is the method that is returned (and that runs) every time 'svgStreetside()' is called.
             * 'svgStreetside()' is called from index.js
             */
            function drawImages(selection) {
 
                function addChildVertices(entity) {
 
-                   // avoid redunant work and infinite recursion of circular relations
+                   // avoid redundant work and infinite recursion of circular relations
                    if (seenIds[entity.id]) { return; }
                    seenIds[entity.id] = true;
 
                }
 
                // Collect important vertices from the `entities` list..
-               // (during a paritial redraw, it will not contain everything)
+               // (during a partial redraw, it will not contain everything)
                for (var i = 0; i < entities.length; i++) {
                    var entity = entities[i];
                    var geometry = entity.geometry(graph);
                        );
 
                        // On Firefox Windows and Linux we always get +/- the scroll line amount (default 3)
-                       // There doesn't seem to be any scroll accelleration.
+                       // There doesn't seem to be any scroll acceleration.
                        // This multiplier increases the speed a little bit - #5512
                        if (detected.os !== 'mac') {
                            dY *= 5;
                        .selectAll('.background-info-span-vintage')
                        .text(metadata.vintage);
 
-                   // update other metdata
+                   // update other metadata
                    metadataKeys.forEach(function(k) {
                        if (k === 'zoom' || k === 'vintage') { return; }  // done already
                        var val = result[k];
                        .preset(_presets.length === 1 ? _presets[0] : _mainPresetIndex.item('point'))
                    );
 
-               // NOTE: split on en-dash, not a hypen (to avoid conflict with hyphenated names)
+               // NOTE: split on en-dash, not a hyphen (to avoid conflict with hyphenated names)
                var names = _presets.length === 1 ? _presets[0].name().split(' – ') : [_t('inspector.multiple_types')];
 
                var label = selection.select('.label-inner');
                if (!utilArrayIdentical(val, _presets)) {
                    _presets = val;
 
-                   var geometries = entityGeometries();
-                   if (_presets.length === 1 && geometries.length) {
-                       _tagReference = uiTagReference(_presets[0].reference(geometries[0]))
+                   if (_presets.length === 1) {
+                       _tagReference = uiTagReference(_presets[0].reference())
                            .showing(false);
                    }
                }
                }
 
                // Remove whatever is after the last ' – '
-               // NOTE: split/join on en-dash, not a hypen (to avoid conflict with fr - nl names in Brussels etc)
+               // NOTE: split/join on en-dash, not a hyphen (to avoid conflict with fr - nl names in Brussels etc)
                function cleanName(name) {
                    var parts = name.split(' – ');
                    if (parts.length > 1) {
              .call(langCombo)
              .merge(_langInput);
 
-           utilGetSetValue(_langInput, language()[1]);
-
            _langInput
              .on('blur', changeLang)
              .on('change', changeLang);
          }
 
 
-         function language() {
+         function defaultLanguageInfo(skipEnglishFallback) {
+           var langCode = _mainLocalizer.languageCode().toLowerCase();
+
+           for (var i in _dataWikipedia) {
+             var d = _dataWikipedia[i];
+             // default to the language of iD's current locale
+             if (d[2] === langCode) { return d; }
+           }
+
+           // fallback to English
+           return skipEnglishFallback ? ['', '', ''] : ['English', 'English', 'en'];
+         }
+
+
+         function language(skipEnglishFallback) {
            var value = utilGetSetValue(_langInput).toLowerCase();
-           var locale = _mainLocalizer.localeCode().toLowerCase();
-           var localeLanguage;
-           return _dataWikipedia.find(function (d) {
-             if (d[2] === locale) { localeLanguage = d; }
-             return d[0].toLowerCase() === value || d[1].toLowerCase() === value || d[2] === value;
-           }) || localeLanguage || ['English', 'English', 'en'];
+
+           for (var i in _dataWikipedia) {
+             var d = _dataWikipedia[i];
+             // return the language already set in the UI, if supported
+             if (d[0].toLowerCase() === value ||
+               d[1].toLowerCase() === value ||
+               d[2] === value) { return d; }
+           }
+
+           // fallback to English
+           return defaultLanguageInfo(skipEnglishFallback);
          }
 
 
          function change(skipWikidata) {
            var value = utilGetSetValue(_titleInput);
            var m = value.match(/https?:\/\/([-a-z]+)\.wikipedia\.org\/(?:wiki|\1-[-a-z]+)\/([^#]+)(?:#(.+))?/);
-           var l = m && _dataWikipedia.find(function (d) { return m[1] === d[2]; });
+           var langInfo = m && _dataWikipedia.find(function (d) { return m[1] === d[2]; });
            var syncTags = {};
 
-           if (l) {
+           if (langInfo) {
+             var nativeLangName = langInfo[1];
              // Normalize title http://www.mediawiki.org/wiki/API:Query#Title_normalization
              value = decodeURIComponent(m[2]).replace(/_/g, ' ');
              if (m[3]) {
                value += '#' + anchor.replace(/_/g, ' ');
              }
              value = value.slice(0, 1).toUpperCase() + value.slice(1);
-             utilGetSetValue(_langInput, l[1]);
+             utilGetSetValue(_langInput, nativeLangName);
              utilGetSetValue(_titleInput, value);
            }
 
          function updateForTags(tags) {
 
            var value = typeof tags[field.key] === 'string' ? tags[field.key] : '';
+           // Expect tag format of `tagLang:tagArticleTitle`, e.g. `fr:Paris`, with
+           // optional suffix of `#anchor`
            var m = value.match(/([^:]+):([^#]+)(?:#(.+))?/);
-           var l = m && _dataWikipedia.find(function (d) { return m[1] === d[2]; });
+           var tagLang = m && m[1];
+           var tagArticleTitle = m && m[2];
            var anchor = m && m[3];
+           var tagLangInfo = tagLang && _dataWikipedia.find(function (d) { return tagLang === d[2]; });
 
            // value in correct format
-           if (l) {
-             utilGetSetValue(_langInput, l[1]);
-             utilGetSetValue(_titleInput, m[2] + (anchor ? ('#' + anchor) : ''));
+           if (tagLangInfo) {
+             var nativeLangName = tagLangInfo[1];
+             utilGetSetValue(_langInput, nativeLangName);
+             utilGetSetValue(_titleInput, tagArticleTitle + (anchor ? ('#' + anchor) : ''));
              if (anchor) {
                try {
                  // Best-effort `anchorencode:` implementation
                  anchor = anchor.replace(/ /g, '_');
                }
              }
-             _wikiURL = 'https://' + m[1] + '.wikipedia.org/wiki/' +
-               m[2].replace(/ /g, '_') + (anchor ? ('#' + anchor) : '');
+             _wikiURL = 'https://' + tagLang + '.wikipedia.org/wiki/' +
+               tagArticleTitle.replace(/ /g, '_') + (anchor ? ('#' + anchor) : '');
 
            // unrecognized value format
            } else {
              utilGetSetValue(_titleInput, value);
              if (value && value !== '') {
                utilGetSetValue(_langInput, '');
-               _wikiURL = "https://en.wikipedia.org/wiki/Special:Search?search=" + value;
+               var defaultLangInfo = defaultLanguageInfo();
+               _wikiURL = "https://" + (defaultLangInfo[2]) + ".wikipedia.org/w/index.php?fulltext=1&search=" + value;
              } else {
+               var shownOrDefaultLangInfo = language(true /* skipEnglishFallback */);
+               utilGetSetValue(_langInput, shownOrDefaultLangInfo[1]);
                _wikiURL = '';
              }
            }
                    var graph = context.graph();
 
                    var geometries = Object.keys(_entityIDs.reduce(function(geoms, entityID) {
-                       return geoms[graph.entity(entityID).geometry(graph)] = true;
+                       geoms[graph.entity(entityID).geometry(graph)] = true;
+                       return geoms;
                    }, {}));
 
                    var presetsManager = _mainPresetIndex;
 
 
            function changeRole(d) {
-               if (d === 0) { return; }    // called on newrow (shoudn't happen)
+               if (d === 0) { return; }    // called on newrow (shouldn't happen)
                if (_inChange) { return; }  // avoid accidental recursive call #5731
 
                var oldRole = d.member.role;
 
            function deleteMembership(d) {
                this.blur();           // avoid keeping focus on the button
-               if (d === 0) { return; }   // called on newrow (shoudn't happen)
+               if (d === 0) { return; }   // called on newrow (shouldn't happen)
 
                // remove the hover-highlight styling
                utilHighlightEntities([d.relation.id], false, context);
                        .append('div')
                        .attr('class', 'label-inner');
 
-                   // NOTE: split/join on en-dash, not a hypen (to avoid conflict with fr - nl names in Brussels etc)
+                   // NOTE: split/join on en-dash, not a hyphen (to avoid conflict with fr - nl names in Brussels etc)
                    label.selectAll('.namepart')
                        .data(preset.name().split(' – '))
                        .enter()
                };
 
                item.preset = preset;
-               item.reference = uiTagReference(preset.reference(entityGeometries()[0]));
+               item.reference = uiTagReference(preset.reference());
 
                return item;
            }
                            .classed('inspector-hidden', false)
                            .classed('inspector-hover', false);
 
-                       if (!inspector.entityIDs() || !utilArrayIdentical(inspector.entityIDs(), ids) || inspector.state() !== 'select') {
-                           inspector
-                               .state('select')
-                               .entityIDs(ids)
-                               .newFeature(newFeature);
+                       // reload the UI even if the ids are the same since the entities
+                       // themselves may have changed
+                       inspector
+                           .state('select')
+                           .entityIDs(ids)
+                           .newFeature(newFeature);
 
-                           inspectorWrap
-                               .call(inspector);
-                       }
+                       inspectorWrap
+                           .call(inspector);
 
                    } else {
                        inspector
          var context = utilRebind({}, dispatch$1, 'on');
          var _deferred = new Set();
 
-         context.version = '2.18.4';
+         context.version = '2.18.5';
          context.privacyVersion = '20200407';
 
          // iD will alter the hash so cache the parameters intended to setup the session